cssutils-1.0/0000755000175000017500000000000012416502415011711 5ustar hugohugocssutils-1.0/COPYING0000666000175000017500000010575512126054542012766 0ustar hugohugo 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 . cssutils-1.0/PKG-INFO0000666000175000017500000002365412253307530013024 0ustar hugohugoMetadata-Version: 1.1 Name: cssutils Version: 1.0 Summary: A CSS Cascading Style Sheets library for Python Home-page: http://cthedot.de/cssutils/ Author: Christof Hoeke Author-email: c@cthedot.de License: LGPL 2.1 or later, see also http://cthedot.de/cssutils/ Download-URL: https://bitbucket.org/cthedot/cssutils/downloads Description: ====== README ====== .. -*- restructuredtext -*- ------------------------------------------------------- cssutils: CSS Cascading Style Sheets library for Python ------------------------------------------------------- :Copyright: 2004-2013 Christof Hoeke Overview ======== A Python package to parse and build CSS Cascading Style Sheets. DOM only, not any rendering facilities! Based upon and partly implementing the following specifications : `CSS 2.1rev1 `__ General CSS rules and properties are defined here `CSS3 Module: Syntax `__ Used in parts since cssutils 0.9.4. cssutils tries to use the features from CSS 2.1 and CSS 3 with preference to CSS3 but as this is not final yet some parts are from CSS 2.1 `CSS Fonts Module Level 3 `__ Added changes and additional stuff (since cssutils v0.9.6) `MediaQueries `__ MediaQueries are part of ``stylesheets.MediaList`` since v0.9.4, used in @import and @media rules. `Namespaces `__ Added in v0.9.1, updated to definition in CSSOM in v0.9.4, updated in 0.9.5 for dev version `CSS3 Module: Pages Media `__ Most properties of this spec are implemented including MarginRules `Selectors `__ The selector syntax defined here (and not in CSS 2.1) should be parsable with cssutils (*should* mind though ;) ) `CSS Backgrounds and Borders Module Level 3 `__, `CSS3 Basic User Interface Module `__, `CSS Text Level 3 `__ Some validation for properties included, mainly `cursor`, `outline`, `resize`, `box-shadow`, `text-shadow` `Variables `__ / `CSS Custom Properties `__ Experimental specification of CSS Variables which cssutils implements partly. The vars defined in the newer CSS Custom Properties spec should in main parts be at least parsable with cssutils. `DOM Level 2 Style CSS `__ DOM for package css. 0.9.8 removes support for CSSValue and related API, see PropertyValue and Value API for now `DOM Level 2 Style Stylesheets `__ DOM for package stylesheets `CSSOM `__ A few details (mainly the NamespaceRule DOM) are taken from here. Plan is to move implementation to the stuff defined here which is newer but still no REC so might change anytime... The cssutils tokenizer is a customized implementation of `CSS3 Module: Syntax (W3C Working Draft 13 August 2003) `_ which itself is based on the CSS 2.1 tokenizer. It tries to be as compliant as possible but uses some (helpful) parts of the CSS 2.1 tokenizer. I guess cssutils is neither CSS 2.1 nor CSS 3 compliant but tries to at least be able to parse both grammars including some more real world cases (some CSS hacks are actually parsed and serialized). Both official grammars are not final nor bugfree but still feasible. cssutils aim is not to be fully compliant to any CSS specification (the specifications seem to be in a constant flow anyway) but cssutils *should* be able to read and write as many as possible CSS stylesheets "in the wild" while at the same time implement the official APIs which are well documented. Some minor extensions are provided as well. Please visit http://cthedot.de/cssutils/ or https://bitbucket.org/cthedot/cssutils/ for more details. There is also a low-traffic `cssutils discussion group `_. Compatibility ============= cssutils is developed on standard Python but works under Python 2.x (from 2.5, 2.7.6 tested), 3.x (v3.3.3 tested) and Jython (from 2.5.1). IronPython has not been tested yet but might work? Python 2.4 and older are not supported since cssutils 0.9.8 anymore. cssutils is not thread safe, please beware! License ======= Copyright 2005 - 2013 Christof Hoeke cssutils is published under the LGPL 3 or later cssutils is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. cssutils 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with cssutils. If not, see http://www.gnu.org/licenses. Installation ============ From 0.9.6 cssutils uses `Distribute `_ After installing Distribute use:: > easy_install cssutils to install the latest version of cssutils. Alternatively download the provided source distribution. Expand the file and from a command line install with:: > python setup.py install To uninstall remove any registrations of cssutils eggs with Distribute and remove the eggs which should be installed at PYTHONDIR/Lib/site-packages/cssutils too. Example ======= :: # -*- coding: utf-8 -*- import cssutils css = u'''/* a comment with umlaut ä */ @namespace html "http://www.w3.org/1999/xhtml"; @variables { BG: #fff } html|a { color:red; background: var(BG) }''' sheet = cssutils.parseString(css) for rule in sheet: if rule.type == rule.STYLE_RULE: # find property for property in rule.style: if property.name == 'color': property.value = 'green' property.priority = 'IMPORTANT' break # or simply: rule.style['margin'] = '01.0eM' # or: ('1em', 'important') sheet.encoding = 'ascii' sheet.namespaces['xhtml'] = 'http://www.w3.org/1999/xhtml' sheet.namespaces['atom'] = 'http://www.w3.org/2005/Atom' sheet.add('atom|title {color: #000000 !important}') sheet.add('@import "sheets/import.css";') # cssutils.ser.prefs.resolveVariables == True since 0.9.7b2 print sheet.cssText results in:: @charset "ascii"; @import "sheets/import.css"; /* a comment with umlaut \E4 */ @namespace xhtml "http://www.w3.org/1999/xhtml"; @namespace atom "http://www.w3.org/2005/Atom"; xhtml|a { color: green !important; background: #fff; margin: 1em } atom|title { color: #000 !important } Documentation ============= The current documenation can be found at http://packages.python.org/cssutils/ Kind Request ============ cssutils is far from being perfect or even complete. If you find any bugs (especially specification violations) or have problems or suggestions please put them in the `Issue Tracker `_ at Bitbucket. Thanks ====== Thanks to Simon Sapin, Jason R. Coombs and Walter Doerwald for patches, help and discussion. Thanks to Kevin D. Smith for the value validating module. Thanks also to Cory Dodt, Tim Gerla, James Dobson and Amit Moscovich for helpful suggestions and code patches. Thanks to Fredrik Hedman for help on port of encutils to Python 3. Keywords: CSS,Cascading Style Sheets,CSSParser,DOM Level 2 Stylesheets,DOM Level 2 CSS Platform: Python 2.5 and later. Python 3.2 and later. Jython 2.5.1 and later. Classifier: Development Status :: 4 - Beta Classifier: Environment :: Web Environment Classifier: Intended Audience :: Developers Classifier: License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL) Classifier: Operating System :: OS Independent Classifier: Programming Language :: Python Classifier: Programming Language :: Python :: 2.5 Classifier: Programming Language :: Python :: 2.6 Classifier: Programming Language :: Python :: 2.7 Classifier: Programming Language :: Python :: 3.2 Classifier: Programming Language :: Python :: 3.3 Classifier: Topic :: Internet Classifier: Topic :: Software Development :: Libraries :: Python Modules Classifier: Topic :: Text Processing :: Markup :: HTML cssutils-1.0/setup.py0000666000175000017500000000573212126104566013441 0ustar hugohugo# -*- coding: utf-8 -*- #!/usr/bin/env python """ cssutils setup use EasyInstall or install with >python setup.py install """ __docformat__ = 'restructuredtext' __author__ = 'Christof Hoeke with contributions by Walter Doerwald and lots of other people' __date__ = '$LastChangedDate:: $:' import codecs import os # For Python 2.5 try: next except NameError: next = lambda iter: iter.next() # extract the version without importing the module lines = open('src/cssutils/__init__.py') is_ver_line = lambda line: line.startswith('VERSION = ') line = next(line for line in lines if is_ver_line(line)) exec(line, locals(), globals()) # use the build_py_2to3 if we're building on Python 3 try: from distutils.command.build_py import build_py_2to3 as build_py except ImportError: from distutils.command.build_py import build_py try: from setuptools import setup, find_packages except ImportError: from ez_setup import use_setuptools use_setuptools() from setuptools import setup, find_packages def read(*rnames): return codecs.open(os.path.join(*rnames), encoding='utf-8').read() long_description = '\n' + read('README.txt') + '\n'# + read('CHANGELOG.txt') setup( name='cssutils', version=VERSION, package_dir={'':'src'}, packages=find_packages('src'), test_suite='cssutils.tests', #'nose.collector' tests_require='mock', entry_points={ 'console_scripts': [ 'csscapture = cssutils.scripts.csscapture:main', 'csscombine = cssutils.scripts.csscombine:main', 'cssparse = cssutils.scripts.cssparse:main' ] }, description='A CSS Cascading Style Sheets library for Python', long_description=long_description, author='Christof Hoeke', author_email='c@cthedot.de', url='http://cthedot.de/cssutils/', download_url='https://bitbucket.org/cthedot/cssutils/downloads', license='LGPL 2.1 or later, see also http://cthedot.de/cssutils/', keywords='CSS, Cascading Style Sheets, CSSParser, DOM Level 2 Stylesheets, DOM Level 2 CSS', platforms='Python 2.5 and later. Python 3.2 and later. Jython 2.5.1 and later.', cmdclass=dict( # specify the build_py command imported earlier build_py=build_py, ), classifiers=[ 'Development Status :: 4 - Beta', 'Environment :: Web Environment', 'Intended Audience :: Developers', 'License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)', 'Operating System :: OS Independent', 'Programming Language :: Python', 'Programming Language :: Python :: 2.5', 'Programming Language :: Python :: 2.6', 'Programming Language :: Python :: 2.7', 'Programming Language :: Python :: 3.2', 'Programming Language :: Python :: 3.3', 'Topic :: Internet', 'Topic :: Software Development :: Libraries :: Python Modules', 'Topic :: Text Processing :: Markup :: HTML' ] ) cssutils-1.0/setup3.py0000666000175000017500000000553212126056220013513 0ustar hugohugo# -*- coding: utf-8 -*- #!/usr/bin/env python """ cssutils setup use EasyInstall or install with >python setup.py install """ __docformat__ = 'restructuredtext' __author__ = 'Christof Hoeke with contributions by Walter Doerwald' __date__ = '$LastChangedDate:: $:' import codecs import os import sys # For Python 2.5 try: next except NameError: next = lambda iter: iter.next # extract the version without importing the module lines = open('src3/cssutils/__init__.py') is_ver_line = lambda line: line.startswith('VERSION = ') line = next(line for line in lines if is_ver_line(line)) # For Python 2.5 try: exec(line, locals(), globals()) except TypeError: exec(line(), locals(), globals()) # use the build_py_2to3 if we're building on Python 3 try: from distutils.command.build_py import build_py_2to3 as build_py except ImportError: from distutils.command.build_py import build_py try: from setuptools import setup, find_packages except ImportError: from ez_setup import use_setuptools use_setuptools() from setuptools import setup, find_packages def read(*rnames): return codecs.open(os.path.join(*rnames), encoding='utf-8').read() long_description = '\n' + read('README.txt') + '\n'# + read('CHANGELOG.txt') setup( name='cssutils', version=VERSION, package_dir={'':'src3'}, packages=find_packages('src3'), test_suite='cssutils.tests', #'nose.collector' tests_require='mock', entry_points={ 'console_scripts': [ 'csscapture = cssutils.scripts.csscapture:main', 'csscombine = cssutils.scripts.csscombine:main', 'cssparse = cssutils.scripts.cssparse:main' ] }, description='A CSS Cascading Style Sheets library for Python', long_description=long_description, author='Christof Hoeke', author_email='c@cthedot.de', url='http://cthedot.de/cssutils/', download_url='http://code.google.com/p/cssutils/downloads/list', license='LGPL 2.1 or later, see also http://cthedot.de/cssutils/', keywords='CSS, Cascading Style Sheets, CSSParser, DOM Level 2 Stylesheets, DOM Level 2 CSS', platforms='Python 3.2 and later.', cmdclass=dict( # specify the build_py command imported earlier build_py=build_py, ), classifiers=[ 'Development Status :: 4 - Beta', 'Environment :: Web Environment', 'Intended Audience :: Developers', 'License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)', 'Operating System :: OS Independent', 'Programming Language :: Python', 'Programming Language :: Python :: 2', 'Programming Language :: Python :: 3', 'Topic :: Internet', 'Topic :: Software Development :: Libraries :: Python Modules', 'Topic :: Text Processing :: Markup :: HTML' ] ) cssutils-1.0/sheets/0000755000175000017500000000000012416502415013204 5ustar hugohugocssutils-1.0/sheets/atrule.css0000666000175000017500000000021112126054542015211 0ustar hugohugo@charset "UTF-8"; @three-dee { @background-lighting { azimuth: 30deg; elevation: 190deg; } H1 { color: red } } @page { foo: bar } cssutils-1.0/sheets/1import.css0000666000175000017500000000006712126054542015321 0ustar hugohugo@charset "iso-8859-1"; @import "1inherit-iso.css"; cssutils-1.0/sheets/var/0000755000175000017500000000000012416502415013774 5ustar hugohugocssutils-1.0/sheets/var/start.css0000666000175000017500000000020512126054542015645 0ustar hugohugo@import "vars.css"; @import "vars2.css"; @import "use.css"; @variables { TEST: 1px; T2: 'T2' } a { left: var(T2); } cssutils-1.0/sheets/var/use.css0000666000175000017500000000044212126054542015307 0ustar hugohugoa { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } a { content: var(TEST) } cssutils-1.0/sheets/var/vars2.css0000666000175000017500000000004112126054542015543 0ustar hugohugo@variables { TEST: 'VARS2' } cssutils-1.0/sheets/var/vars.css0000666000175000017500000000005112126054542015462 0ustar hugohugo@variables { TEST: 1px; T2: 'T2' } cssutils-1.0/sheets/acid2.css0000666000175000017500000002243612126054542014714 0ustar hugohugo /* section numbers refer to CSS2.1 */ /* page setup */ html { font: 12px sans-serif; margin: 0; padding: 0; overflow: hidden; /* hides scrollbars on viewport, see 11.1.1:3 */ background: white; color: red; } body { margin: 0; padding: 0; } /* introduction message */ .intro { font: 2em sans-serif; margin: 3.5em 2em; padding: 0.5em; border: solid thin; background: white; color: black; position: relative; z-index: 2; /* should cover the black and red bars that are fixed-positioned */ } .intro * { font: inherit; margin: 0; padding: 0; } .intro h1 { font-size: 1em; font-weight: bolder; margin: 0; padding: 0; } .intro :link { color: blue; } .intro :visited { color: purple; } /* picture setup */ #top { margin: 100em 3em 0; padding: 2em 0 0 .5em; text-align: left; font: 2em/24px sans-serif; color: navy; white-space: pre; } /* "Hello World!" text */ .picture { position: relative; border: 1em solid transparent; margin: 0 0 100em 3em; } /* containing block for face */ .picture { background: red; } /* overriden by preferred stylesheet below */ /* top line of face (scalp): fixed positioning and min/max height/width */ .picture p { position: fixed; margin: 0; padding: 0; border: 0; top: 9em; left: 11em; width: 140%; max-width: 4em; height: 8px; min-height: 1em; max-height: 2mm; /* min-height overrides max-height, see 10.7 */ background: black; border-bottom: 0.5em yellow solid; } /* bits that shouldn't be part of the top line (and shouldn't be visible at all): HTML parsing, "+" combinator, stacking order */ .picture p.bad { border-bottom: red solid; /* shouldn't matter, because the "p + table + p" rule below should match it too, thus hiding it */ } .picture p + p { background: maroon; z-index: 1; } /* shouldn't match anything */ .picture p + table + p { margin-top: 3em; /* should end up under the absolutely positioned table below, and thus not be visible */ } /* second line of face: attribute selectors, float positioning */ [class~=one].first.one { position: absolute; top: 0; margin: 36px 0 0 60px; padding: 0; border: black 2em; border-style: none solid; /* shrink wraps around float */ } [class~=one][class~=first] [class=second\ two][class="second two"] { float: right; width: 48px; height: 12px; background: yellow; margin: 0; padding: 0; } /* only content of abs pos block */ /* third line of face: width and overflow */ .forehead { margin: 4em; width: 8em; border-left: solid black 1em; border-right: solid black 1em; background: red url(%2F58BAAT%2FAf9jgNErAAAAAElFTkSuQmCC); /* that's a 1x1 yellow pixel PNG */ } .forehead * { width: 12em; line-height: 1em; } /* class selectors headache */ .two.error.two { background: maroon; } /* shouldn't match */ .forehead.error.forehead { background: red; } /* shouldn't match */ [class=second two] { background: red; } /* this should be ignored (invalid selector -- grammar says it only accepts IDENTs or STRINGs) */ /* fourth and fifth lines of face, with eyes: paint order test (see appendix E) and fixed backgrounds */ /* the two images are identical: 2-by-2 squares with the top left and bottom right pixels set to yellow and the other two set to transparent. Since they are offset by one pixel from each other, the second one paints exactly over the transparent parts of the first one, thus creating a solid yellow block. */ .eyes { position: absolute; top: 5em; left: 3em; margin: 0; padding: 0; background: red; } #eyes-a { height: 0; line-height: 2em; text-align: right; } /* contents should paint top-most because they're inline */ #eyes-a object { display: inline; vertical-align: bottom; } #eyes-a object[type] { width: 7.5em; height: 2.5em; } /* should have no effect since that object should fallback to being inline (height/width don't apply to inlines) */ #eyes-a object object object { border-right: solid 1em black; padding: 0 12px 0 11px; background: url(%2FwD%2FAP%2BgvaeTAAAAEUlEQVR42mP4%2F58BCv7%2FZwAAHfAD%2FabwPj4AAAAASUVORK5CYII%3D) fixed 1px 0; } #eyes-b { float: left; width: 10em; height: 2em; background: fixed url(%2FwD%2FAP%2BgvaeTAAAAEUlEQVR42mP4%2F58BCv7%2FZwAAHfAD%2FabwPj4AAAAASUVORK5CYII%3D); border-left: solid 1em black; border-right: solid 1em red; } /* should paint in the middle layer because it is a float */ #eyes-c { display: block; background: red; border-left: 2em solid yellow; width: 10em; height: 2em; } /* should paint bottom most because it is a block */ /* lines six to nine, with nose: auto margins */ .nose { float: left; margin: -2em 2em -1em; border: solid 1em black; border-top: 0; min-height: 80%; height: 60%; max-height: 3em; /* percentages become auto (see 10.5 and 10.7) and intrinsic height is more than 3em, so 3em wins */ padding: 0; width: 12em; } .nose > div { padding: 1em 1em 3em; height: 0; background: yellow; } .nose div div { width: 2em; height: 2em; background: red; margin: auto; } .nose :hover div { border-color: blue; } .nose div:hover :before { border-bottom-color: inherit; } .nose div:hover :after { border-top-color: inherit; } .nose div div:before { display: block; border-style: none solid solid; border-color: red yellow black yellow; border-width: 1em; content: ''; height: 0; } .nose div :after { display: block; border-style: solid solid none; border-color: black yellow red yellow; border-width: 1em; content: ''; height: 0; } /* between lines nine and ten: margin collapsing with 'float' and 'clear' */ .empty { margin: 6.25em; height: 10%; /* computes to auto which makes it empty per 8.3.1:7 (own margins) */ } .empty div { margin: 0 2em -6em 4em; } .smile { margin: 5em 3em; clear: both; /* clearance is negative (see 8.3.1 and 9.5.1) */ } /* line ten and eleven: containing block for abs pos */ .smile div { margin-top: 0.25em; background: black; width: 12em; height: 2em; position: relative; bottom: -1em; } .smile div div { position: absolute; top: 0; right: 1em; width: auto; height: 0; margin: 0; border: yellow solid 1em; } /* smile (over lines ten and eleven): backgrounds behind borders, inheritance of 'float', nested floats, negative heights */ .smile div div span { display: inline; margin: -1em 0 0 0; border: solid 1em transparent; border-style: none solid; float: right; background: black; height: 1em; } .smile div div span em { float: inherit; border-top: solid yellow 1em; border-bottom: solid black 1em; } /* zero-height block; width comes from (zero-height) child. */ .smile div div span em strong { width: 6em; display: block; margin-bottom: -1em; /* should have no effect, since parent has top&bottom borders, so this margin doesn't collapse */ } /* line twelve: line-height */ .chin { margin: -4em 4em 0; width: 8em; line-height: 1em; border-left: solid 1em black; border-right: solid 1em black; background: yellow url(%2F%2F6wf8CJBJTK9lnQ7FpHGaOurt1I34nfH9pMMZAZ8BwMGEvvh%2BBsJCAgICLwIOA8EBAQEBAQEBAQEBK79H5RfIQAAAAAAAAAAAAAAAAAAAAAAAAAAAID%2FABMSqAfj%2FsLmvAAAAABJRU5ErkJggg%3D%3D) /* 64x64 red square */ no-repeat fixed /* shouldn't be visible unless the smiley is moved to the top left of the viewport */; } .chin div { display: inline; font: 2px/4px serif; } /* line thirteen: cascade and selector tests */ .parser-container div { color: maroon; border: solid; color: orange; } /* setup */ div.parser-container * { border-color: black; /* overrides (implied) border-color on previous line */ } /* setup */ * div.parser { border-width: 0 2em; /* overrides (implied) declarations on earlier line */ } /* setup */ /* line thirteen continued: parser tests */ .parser { /* comment parsing test -- comment ends before the end of this line, the backslash should have no effect: \*/ } .parser { margin: 0 5em 1em; padding: 0 1em; width: 2em; height: 1em; error: \}; background: yellow; } /* setup with parsing test */ * html .parser { background: gray; } \.parser { padding: 2em; } .parser { m\argin: 2em; }; .parser { height: 3em; } .parser { width: 200; } .parser { border: 5em solid red ! error; } .parser { background: red pink; } /* line fourteen (last line of face): table */ ul { display: table; padding: 0; margin: -1em 7em 0; background: red; } ul li { padding: 0; margin: 0; } ul li.first-part { display: table-cell; height: 1em; width: 1em; background: black; } ul li.second-part { display: table; height: 1em; width: 1em; background: black; } /* anonymous table cell wraps around this */ ul li.third-part { display: table-cell; height: 0.5em; /* gets stretched to fit row */ width: 1em; background: black; } ul li.fourth-part { list-style: none; height: 1em; width: 1em; background: black; } /* anonymous table cell wraps around this */ /* bits that shouldn't appear: inline alignment in cells */ .image-height-test { height: 10px; overflow: hidden; font: 20em serif; } /* only the area between the top of the line box and the top of the image should be visible */ table { margin: 0; border-spacing: 0; } td { padding: 0; }cssutils-1.0/sheets/csscombine-proxy.css0000666000175000017500000000033412126054542017227 0ustar hugohugo@charset "utf-8"; /* proxy sheet were imported sheets should be combined */ /* non-ascii chars: öäü */ @import "csscombine-1.css"; @import url(csscombine-2.css); @namespace other "other"; proxy { top: 3px } cssutils-1.0/sheets/html40.css0000666000175000017500000001026212126054542015034 0ustar hugohugo@charset "US-ASCII"; ADDRESS, BLOCKQUOTE, BODY, DD, DIV, DL, DT, FIELDSET, FORM, FRAME, FRAMESET, H1, H2, H3, H4, H5, H6, IFRAME, NOFRAMES, OBJECT, OL, P, UL, APPLET, CENTER, DIR, HR, MENU, PRE { display: block } LI { display: list-item } HEAD { display: none } TABLE { display: table } TR { display: table-row } THEAD { display: table-header-group } TBODY { display: table-row-group } TFOOT { display: table-footer-group } COL { display: table-column } COLGROUP { display: table-column-group } TD, TH { display: table-cell } CAPTION { display: table-caption } TH { font-weight: bolder; text-align: center } CAPTION { text-align: center } BODY { padding: 8px; line-height: 1.33 } H1 { font-size: 2em; margin: .67em 0 } H2 { font-size: 1.5em; margin: .83em 0 } H3 { font-size: 1.17em; margin: 1em 0 } H4, P, BLOCKQUOTE, UL, FIELDSET, FORM, OL, DL, DIR, MENU { margin: 1.33em 0 } H5 { font-size: .83em; line-height: 1.17em; margin: 1.67em 0 } H6 { font-size: .67em; margin: 2.33em 0 } H1, H2, H3, H4, H5, H6, B, STRONG { font-weight: bolder } BLOCKQUOTE { margin-left: 40px; margin-right: 40px } I, CITE, EM, VAR, ADDRESS { font-style: italic } PRE, TT, CODE, KBD, SAMP { font-family: monospace } PRE { white-space: pre } BIG { font-size: 1.17em } SMALL, SUB, SUP { font-size: .83em } SUB { vertical-align: sub } SUP { vertical-align: super } S, STRIKE, DEL { text-decoration: line-through } HR { border: 1px inset } OL, UL, DIR, MENU, DD { margin-left: 40px } OL { list-style-type: decimal } OL UL, UL OL, UL UL, OL OL { margin-top: 0; margin-bottom: 0 } U, INS { text-decoration: underline } CENTER { text-align: center } BR:before { content: "\A" } /* An example of style for HTML 4.0's ABBR/ACRONYM elements */ ABBR, ACRONYM { font-variant: small-caps; letter-spacing: 0.1em } A[href] { text-decoration: underline } :focus { outline: thin dotted invert } /* Begin bidirectionality settings (do not change) */ BDO[DIR="ltr"] { direction: ltr; unicode-bidi: bidi-override } BDO[DIR="rtl"] { direction: rtl; unicode-bidi: bidi-override } *[DIR="ltr"] { direction: ltr; unicode-bidi: embed } *[DIR="rtl"] { direction: rtl; unicode-bidi: embed } /* Elements that are block-level in HTML4 */ ADDRESS, BLOCKQUOTE, BODY, DD, DIV, DL, DT, FIELDSET, FORM, FRAME, FRAMESET, H1, H2, H3, H4, H5, H6, IFRAME, NOSCRIPT, NOFRAMES, OBJECT, OL, P, UL, APPLET, CENTER, DIR, HR, MENU, PRE, LI, TABLE, TR, THEAD, TBODY, TFOOT, COL, COLGROUP, TD, TH, CAPTION { unicode-bidi: embed } /* End bidi settings */ @media print { @page { margin: 10% } H1, H2, H3, H4, H5, H6 { page-break-after: avoid; page-break-inside: avoid } BLOCKQUOTE, PRE { page-break-inside: avoid } UL, OL, DL { page-break-before: avoid } } @media speech { H1, H2, H3, H4, H5, H6 { voice-family: paul, male; stress: 20; richness: 90 } H1 { pitch: x-low; pitch-range: 90 } H2 { pitch: x-low; pitch-range: 80 } H3 { pitch: low; pitch-range: 70 } H4 { pitch: medium; pitch-range: 60 } H5 { pitch: medium; pitch-range: 50 } H6 { pitch: medium; pitch-range: 40 } LI, DT, DD { pitch: medium; richness: 60 } DT { stress: 80 } PRE, CODE, TT { pitch: medium; pitch-range: 0; stress: 0; richness: 80 } EM { pitch: medium; pitch-range: 60; stress: 60; richness: 50 } STRONG { pitch: medium; pitch-range: 60; stress: 90; richness: 90 } DFN { pitch: high; pitch-range: 60; stress: 60 } S, STRIKE { richness: 0 } I { pitch: medium; pitch-range: 60; stress: 60; richness: 50 } B { pitch: medium; pitch-range: 60; stress: 90; richness: 90 } U { richness: 0 } A:link { voice-family: harry, male } A:visited { voice-family: betty, female } A:active { voice-family: betty, female; pitch-range: 80; pitch: x-high } } cssutils-1.0/sheets/default_html4.css0000666000175000017500000000535512126054542016467 0ustar hugohugo/* default HTML 4 CSSStyleSheet from http://www.w3.org/TR/2004/CR-CSS21-20040225/sample.html */ html, address, blockquote, body, dd, div, dl, dt, fieldset, form, frame, frameset, h1, h2, h3, h4, h5, h6, noframes, ol, p, ul, center, dir, hr, menu, pre { display: block } li { display: list-item } head { display: none } table { display: table } tr { display: table-row } thead { display: table-header-group } tbody { display: table-row-group } tfoot { display: table-footer-group } col { display: table-column } colgroup { display: table-column-group } td, th { display: table-cell } caption { display: table-caption } th { font-weight: bolder; text-align: center } caption { text-align: center } body { margin: 8px; line-height: 1.12 } h1 { font-size: 2em; margin: .67em 0 } h2 { font-size: 1.5em; margin: .75em 0 } h3 { font-size: 1.17em; margin: .83em 0 } h4, p, blockquote, ul, fieldset, form, ol, dl, dir, menu { margin: 1.12em 0 } h5 { font-size: .83em; margin: 1.5em 0 } h6 { font-size: .75em; margin: 1.67em 0 } h1, h2, h3, h4, h5, h6, b, strong { font-weight: bolder } blockquote { margin-left: 40px; margin-right: 40px } i, cite, em, var, address { font-style: italic } pre, tt, code, kbd, samp { font-family: monospace } pre { white-space: pre } button, textarea, input, object, select { display: inline-block } big { font-size: 1.17em } small, sub, sup { font-size: .83em } sub { vertical-align: sub } sup { vertical-align: super } table { border-spacing: 2px; } thead, tbody, tfoot { vertical-align: middle } td, th { vertical-align: inherit } s, strike, del { text-decoration: line-through } hr { border: 1px inset } ol, ul, dir, menu, dd { margin-left: 40px } ol { list-style-type: decimal } ol ul, ul ol, ul ul, ol ol { margin-top: 0; margin-bottom: 0 } u, ins { text-decoration: underline } br:before { content: "\A" } :before, :after { white-space: pre-line } center { text-align: center } abbr, acronym { font-variant: small-caps; letter-spacing: 0.1em } :link, :visited { text-decoration: underline } :focus { outline: thin dotted invert } /* Begin bidirectionality settings (do not change) */ BDO[DIR="ltr"] { direction: ltr; unicode-bidi: bidi-override } BDO[DIR="rtl"] { direction: rtl; unicode-bidi: bidi-override } *[DIR="ltr"] { direction: ltr; unicode-bidi: embed } *[DIR="rtl"] { direction: rtl; unicode-bidi: embed } @media print { h1 { page-break-before: always } h1, h2, h3, h4, h5, h6 { page-break-after: avoid } ul, ol, dl { page-break-before: avoid } } cssutils-1.0/sheets/cthedot_default.css0000666000175000017500000000741312126054542017066 0ustar hugohugo/* default stylesheet for all variations */ .jsonly { display: none; } .stylenav { text-align: right; margin-top: -1em; } html, body { padding: 0; margin: 0; } body { font: normal 90%/1.1 Georgia, Verdana, "Lucida Grande", Helvetica, sans-serif; color: #fff; background-color: #344; } h1, h2, h3, h4, h5, h6 { font: normal 2.4em Verdana, "Lucida Grande", Arial, Helvetica, sans-serif; } h1, h2, h3, caption { color: #a00; margin: 0.3em -0.4em 0.5em -0.5em; } h2, h3, caption { margin: 0.3em -1.3em 0.3em 0; } h2 { font-size: 1.5em; border-right: 1.3em solid #677; border-bottom: 1px dotted #677; padding-left: 0.1em; padding-bottom: 0.1em; margin-top: 2.1em; margin-bottom: 0.4em; } h3 { font-size: 0.9em; font-weight: bold; text-transform: uppercase; margin-top: 1.2em; margin-bottom: 0.1em; } caption { font-size: 1.05em; text-align: left; margin-bottom: 0.2em; } h4 { font-size: 0.9em; font-weight: bold; margin: 1.2em 0 0; } h5, h6 { font-size: 1em; margin: 0; } h6 { font-size: 0.9em; } p, ol, ul, dl { line-height: 1.3; margin-top: 0; margin-bottom: 1em; } ul { list-style-type: square; } ul.code { line-height: 1.3; } li, dd { margin-bottom: 0.3em; } dt { font-weight: bold; } pre, code { color: #00a; line-height: 1.4; font-size: 1.1em; } table code, table pre { font-size: 1.3em; } .deprecated { color: #888; } table { font-size: 0.9em; border-collapse: collapse; } tr { vertical-align: top; line-height: 1.3; } td, th { text-align: left; padding: 0.4em 0.5em; border-bottom: 1px dotted #667; } td.center { text-align: center; } tr:hover, li:hover { background-color: #f8f8f8; } acronym, .explain { border-bottom: 1px dotted #344; } a { text-decoration: none; color: #fff; border-bottom: 1px solid #aaa; } #main a { color: #a00; } a:visited { color: #eee; } #main a:visited { color: #344; } a:hover { text-decoration: underline; color: #fff; } #main a:hover { background-color: #f5f8ff; } #main a:active { color: #fff; background-color: #abb; } label { display: block; padding: 0.5em 0 0.1em; } input, textarea { font: bold 1em Georgia, Verdana, "Lucida Grande", Helvetica, sans-serif; background-color: #eee; width: 100%; border: 1px inset; } #submit, textarea { margin-bottom: 1.5em; } #submit { font-weight: bold; color: #00a; background-color: #fff; border: 1px outset; margin-top: 2em; } input:focus, input:hover, textarea:focus, textarea:hover { font-weight: bold; background-color: #fff; border-style: solid; } #submit:hover, #submit:focus { background-color: #eee; border-style: solid; } #submit:active { border-style: inset; } #header { padding: 1.1em 5% 0; padding: 40px 5% 0; color: #334; background: #fff url(/img/header_r.jpg) no-repeat; border-bottom: 1px solid #344; height: 90px; he\ight: 50px; } #header dt, #header p { font-family: Arial, Helvetica, sans-serif; letter-spacing: 0.3em; } #header a { color: #334; } #main .nav { padding-bottom: 0.5em; border-bottom: 3px solid #eee; margin-bottom: 1em; margin-left: -8%; } .nav dt , .nav dd, .nav dd ul, .nav li { display: inline; } .nav dt { font-weight: bold; } .nav li { font-weight: bold; padding-right: 0.5em; } .nav a { font-weight: normal; } #footer { padding: 1em 5% 1em 10%; border-top: 3px double #fff; text-align: right; } #main { color: #000; background-color: #fff; padding: 1em 26% 1em 12%; } cssutils-1.0/sheets/ll2.css0000666000175000017500000027731312126054542014431 0ustar hugohugo/* >> Zusaetzliche Steuerelemente (Zurueck, Drucken, Empfehlung versenden) */ /* Steuerelement Container oben*/ div.topAddContrContainer { display: none; margin-top: 5px; } /* Steuerelement Container unten*/ div.bottomAddContrContainer { padding: 10px 15px; } * html div.bottomAddContrContainer { width: 100%; } *+html div.bottomAddContrContainer { width: 100%; } /* Anpassungen der Steuerelemente Tabelle*/ table.aCShell { width: 100%; } table.aCShell td.rSpc { } /* Definiert das Icon fuer Zurueck */ img.backIcon { width: 17px; height: 14px; border: 0; vertical-align: text-top; background: url(../xist4c/web/krusekopf/01/img/backIcon.gif) bottom left no-repeat; } img.recomButton { width: 28px; height: 17px; border: 0; vertical-align: middle; background: url(../xist4c/web/krusekopf/01/img/recommendationButton.gif) bottom right no-repeat; } img.printButton { width: 24px; height: 17px; border: 0; vertical-align: middle; background: url(../xist4c/web/krusekopf/01/img/printIcon.gif) bottom right no-repeat; } img.bFreeButton { width: 24px; height: 18px; border: 0; vertical-align: middle; background: url(../xist4c/web/krusekopf/01/img/barrierFreeIcon.gif) bottom right no-repeat; display: none; } img.cBlindButton, img.cBlindResetButton { width: 26px; height: 18px; border: 0; vertical-align: middle; background: url(../xist4c/web/krusekopf/01/img/colorBlindIcon.gif) center right no-repeat; display: none; } img.cBlindResetButton { background: url(../xist4c/web/krusekopf/01/img/colorBlindResetIcon.gif) center right no-repeat; } /* << */ /* >> Autonews Uebersicht */ /* Autonews Tabelle */ table.aN { } /* Definition fuer das Datum auf der Uebersicht */ table.aN td.date { font-weight: bold; color: #666; padding-right: 8px; } /* Anpassungen der Beschreibungsspalte*/ table.aN td.item { padding-bottom: 10px; } /* Anpassungen des Titels auf der Uebersicht */ table.aN td.item h3 { margin: 0 0 8px 0; } /* Anpassungen des Untertitels auf der Uebersicht */ table.aN td.item h4 { margin: 0 0 3px 0; } /* Definitionen fuer das Uebersichtsbild */ table.aN td.item a img, table.aN td.item img { float: right; margin-left: 10px; margin-bottom: 5px; } /* Anpassungen fuer die Autonews Beschreibung auf der Uebersicht*/ table.aN td.item div.desc { margin: 0; } /* Einstellungen des "mehr..." Links */ table.aN td.item div.more { clear: both; margin-top: 5px; margin-bottom: 5px; } /* Definitionen fuer das "mehr..." Link Icon */ table.aN td.item div.more img { display: none; } /* << */ /* >> Autonews Panel */ /* Auto news Tabelle */ table.aNP { } /* Definition des Autonews Panel Datum */ table.aNP td span.date { font-size: 10px; padding-bottom: 3px; } /* Einstellungen des Autonews Titels */ table.aNP td h3.title { display: inline; margin: 0 0 1px 0; font-size: 12px; } /* Formatiert die Trennlinie zwischen Ueberschrift und Unterueberschrift*/ table.aNP td img.line { width: 100%; margin: 2px 0 2px 0; background: #000; } /* Einstellungen des Autonews Untertitels */ table.aNP td h4.subtitle { margin: 0 0 3px 0; font-size: 11px; } /* Einstellungen der Autonews Beschreibung */ table.aNP td div.desc { margin: 0; font-size: 11px; } /* Einstellungen fuer das Autonewsbild */ table.aNP td img.rFloat, table.aNP td a img.rFloat { float: right; margin-left: 5px; margin-bottom: 2px; } /* Definitionen fuer den "mehr..." Link */ table.aNP td div.more { clear: both; margin-top: 5px; margin-bottom: 5px; font-size: 11px; } /* Definitionen fuer das "mehr..." Link Icon */ table.aNP td div.more img { display: none; } /* Autonews sublink Tabelle */ table.aNPSublink { } /* Definiert den Link mit dem es zu der Autonews Uebersicht geht */ table.aNPSublink td a { font-size: 11px; } /* Anpassen des Autonews Panel "zur Uebersicht...." Icons */ table.aNPSublink td a img { display: none; } /* << */ /* >> Sprungmarke */ /* Sprungmarkencontainer */ div.bToTop { margin-top: 0; margin-bottom: 15px; } /* Sprungmarkencontainer Designelement*/ div.bToTop div.des1 { text-align: right; } /* Sprungmarkencontainer Prompt*/ div.bToTop div.des1 span { font-size: 11px; } /* Sprungmarkencontainer Prompt Link*/ div.bToTop div.des1 span a.text { } /* Sprungmarkencontainer Bild*/ div.bToTop div.des1 img { width: 18px; height: 13px; vertical-align: text-top; background: url(../xist4c/web/krusekopf/01/img/bToTopImg.gif) top right no-repeat; border: 0; } /* Sprungmarkencontainer Bildlink*/ div.bToTop div.des1 a.img { } /* << */ /* >> Standard body Definitionen */ /* Legt Hoehe und Breite fest */ html, body { margin: 0; padding: 0; } /* Anpassungen fuer die Standardeinstellungen des Bodys */ body { background: #fff; font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; font-size: 12px; color: #333; } /* << */ /* >> Angaben nicht veraendern!!! */ /* for framesets */ body.mainFrame { background: #fff; font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; font-size: 12px; color: #000; } /* for print page */ body.print { background: #fff; font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; font-size: 13px; color: #000; } /* << */ /* >> Freie Inhaltscont*/ /* Standard Definition fuer die freien Inhaltscontainer. Diese Angabe bitte unveraendert lassen*/ div.stdSty { height: 100px; width: 100px; position: absolute; left: 0; top: 0; z-index: 0; } /* Hier koennen die angelegten Container definiert werden */ div.banner { width: 468px; height: auto; position: absolute; left: 280px; top: 19px; z-index: 4; display: none; } /* << */ /* >> Dokumentenliste */ /* Document Liste Tabelle */ table.dList { margin: 0 15px; } table.dList a { font-weight: bold; } /* Einstellungen fuer die Dateibeschreibung */ table.dList td span.desc { font-size: 11px; } /* Groesse des Listentitels */ table.dList td div.size { font-size: 11px; color: #000; white-space: nowrap; } /* << */ /* >> Fragen und Antworten*/ /* Einstellungen fuer Gruppenkontainer */ div.faqGrShell { margin-bottom: 15px; } /* Definition der Gruppentitel in der Fragenuebersicht */ div.faqGrShell h3 { margin-bottom: 3px; } /* Anpassungen fuer den Listenelementekontainer */ div.faqGrShell ul { margin-top: 0; margin-bottom: 0; } /* Definitionen des Listenelements */ div.faqGrShell ul li { margin-bottom: 5px; } /* Einstellungen fuer den Fragekontainer in der Detailansicht*/ div.dQShell { margin: 10px 0 15px 0; border: 1px solid #000; background: #dde5ee; } /* Anpassungen des "Frage" Titels auf der Detailseite */ div.dQShell div.prefix { padding: 5px; background: #a0b8cf; font-weight: bold; font-size: 16px; } /* Einstellungen der Detail Ansicht der Frage */ div.dQShell div.question { padding: 5px 5px 5px 20px; } /* << */ /* >> Termin uebersicht */ div.appointmentOuterShell { } div.appointmentOuterShell div.grpShell { margin-top: 10px; margin-bottom: 15px; background: #fff; } div.appointmentOuterShell div.grpShell h3 { font-size: 14px; margin: 0; background: #E3DEC4; padding: 2px 5px 2px 5px; } div.appointmentOuterShell div.grpShell div.item { padding: 4px 5px 4px 5px; border-bottom: 1px solid #dde5ee; } div.appointmentOuterShell div.grpShell div.item div.date { float: left; width: 70px; } div.appointmentOuterShell div.grpShell div.item div.title { margin-left: 80px; margin-bottom: 5px; } div.appointmentOuterShell div.grpShell div.item div.desc { margin-left: 82px; margin-bottom: 5px; font-size: 11px; } div.appointmentOuterShell div.grpShell div.item div.img { float: right; margin-left: 5px; } div.appointmentOuterShell div.grpShell div.item div.more { margin-bottom: 8px; } div.appointmentOuterShell div.grpShell div.item div.more div { text-align: right; } div.appointmentOuterShell div.grpShell div.item div.more div a { background: url(../xist4c/web/krusekopf/01/img/littleTeaserArrow.gif) 0 5px no-repeat; padding-left: 10px; } /* << */ /* >>Erste Ebene Navigation*/ /* Erste Ebene Navigation aeusserer Navigationskontainer*/ div.fLOuterShell { width: 989px; } /* Anpassungen der erste Ebene Navigationspalte */ div.fLOuterShell div.des1 { background: #513E11 url(../xist4c/web/krusekopf/01/img/background.gif) top left repeat-y; } div.fLOuterShell div.des1 div.des2 { } /* Einstellungen fuer die erste Ebene Navigation */ div.fLOuterShell div.des1 table.nav { } /* Spacer fuer die Ausrichtung der erste Ebene Navigation*/ div.fLOuterShell div.des1 table.nav td.spcLeft { } div.fLOuterShell div.des1 table.nav td.spcLeft img { width: 4px; height: 1px; } div.fLOuterShell div.des1 table.nav td.spcRight { } div.fLOuterShell div.des1 table.nav td.spcRight img { width: 4px; height: 1px; } /* Spalte fuer ein erste Ebene Navigationselement */ div.fLOuterShell div.des1 table.nav td.navItem { } /* Grundeinstellung der erste Ebene Navigation fuer die Zustaende "normal", "im Pfad" und "hier"*/ div.fLOuterShell div.des1 table.nav td.navItem span.here, div.fLOuterShell div.des1 table.nav td.navItem a.normal, div.fLOuterShell div.des1 table.nav td.navItem a.inPath { font-size: 12px; color: #000; display: block; width: auto; height: 34px; } div.fLOuterShell div.des1 table.nav td.navItem span.here span, div.fLOuterShell div.des1 table.nav td.navItem a.normal span, div.fLOuterShell div.des1 table.nav td.navItem a.inPath span { display: none; } /* Abweichende einstellungen fuer den Zustand "im Pfad" */ div.fLOuterShell div.des1 table.nav td.navItem a.inPath { font-style: italic; } /* Folgende Angaben definieren die Reaktion beim ueberfahren mit der Maus */ div.fLOuterShell div.des1 table.nav td.navItem a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem a.inPath:hover { color: #336; } /* Angabe fuer den Zustand "hier" */ div.fLOuterShell div.des1 table.nav td.navItem span.here { color: #336; } /* Einstellung der Trennelemente */ div.fLOuterShell div.des1 table.nav td.sep img { height: 34px; background-color: #fff; } /* Kueche und Haushalt */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.inPath { width: 125px; background: url(../xist4c/web/krusekopf/01/img/nav_kueche.gif) top left no-repeat; } *+html div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.normal, *+html div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.normal:hover, *+html div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 span.here, *+html div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.inPath { width: 123px; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav1 a.inPath { background-image: url(../xist4c/web/krusekopf/01/img/nav_kueche2.gif); } /* Haus und Garten */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 a.inPath { width: 110px; background: url(../xist4c/web/krusekopf/01/img/nav_haus.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav2 a.inPath { background-image: url(../xist4c/web/krusekopf/01/img/nav_haus2.gif); } /* Wellness und Beauty */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 a.inPath { width: 135px; background: url(../xist4c/web/krusekopf/01/img/nav_wellness.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav3 a.inPath { background-image: url(../xist4c/web/krusekopf/01/img/nav_wellness2.gif); } /* Arbeit und Buero */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 a.inPath { width: 100px; background: url(../../upload/nav_arbeit_3430.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav4 a.inPath { background-image: url(../../upload/nav_arbeit2_3431.gif); } /* Delikatessen und Suesses */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 a.inPath { width: 154px; background: url(../xist4c/web/krusekopf/01/img/nav_delikatessen.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav5 a.inPath { background-image: url(../xist4c/web/krusekopf/01/img/nav_delikatessen2.gif); } /* Geschenke und Accessoires */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 a.inPath { width: 170px; background: url(.././upload/nav_geschenke_3432.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav6 a.inPath { background-image: url(../../upload/nav_geschenke2_3433.gif); } /* Blumen Online */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 a.inPath { width: 80px; background: url(../../upload/blumen_9221.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav7 a.inPath { background-image: url(../../upload/blumen2_9222.gif); } /* trendmagazin */ div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 a.normal, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 a.inPath { width: 100px; background: url(../xist4c/web/krusekopf/01/img/nav_trend.gif) top left no-repeat; } div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 a.normal:hover, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 span.here, div.fLOuterShell div.des1 table.nav td.navItem div.co_flNav8 a.inPath { background-image: url(../../upload/nav_trend2_2145.gif); } /* << */ /* >> Fusszeile*/ /* Spalte des Footers */ td.msFooter { padding-bottom: 20px; } /* Anpassungen fuer den Fusszeilenkontainer*/ div.decoEl { width: 989px; background: #E3DEC4 url(../xist4c/web/krusekopf/01/img/background.gif) top left repeat-y; } div.deco_1 { background: url(../xist4c/web/krusekopf/01/img/footerbg.gif) bottom left no-repeat; position: relative; top: 1px; } div.deco_2 { } /* Definitionen fuer die Fusszeilentabelle */ table.footer { width: 951px; margin: 0 auto; } /* Einstellungen der Fusszeilen Tabellenzelle */ table.footer td { font-size: 11px; color: #000; text-align: left; padding: 4px 0; } /* Anpassungen der Fusszeilen Inhaltselemente*/ table.footer td p, table.footer td ul, table.footer td ol, table.footer td form, table.footer td h1, table.footer td h2, table.footer td h3, table.footer td h4, table.footer td h5, table.footer td h6 { font-size: 11px; padding-top: 0; padding-bottom: 0; margin-top: 0; margin-bottom: 0; } table.footer td a { color: #5C4714; } table.footer td a:hover { color: #000; } /* << */ /* >>Full Size Media*/ /* Voll Breite Media Container */ div.fsMediaShell { margin-bottom: 10px; } /* Bildelement */ div.fsMediaShell div.img { } /* Freies XHTML Element */ div.fsMediaShell div.free { } /* << */ /* >> Standard Listenelement */ /* Anpassungen fuer den Listeneintragkontainer */ div.gOvItem { margin: 12px 15px; border-bottom: 1px solid #513E11; font-size: 11px; } /* Definition des Titels des Listeneintrags */ div.gOvItem h3 { margin: 0 0 5px 0; } /* Definition des Untertitels des Listeneintrags */ div.gOvItem h4 { margin: 0 0 3px 0; } div.gOvItem p { font-size: 11px; } /* Definition der Beschreibung des Listeneintrags */ div.gOvItem div { margin: 0; } /* Definitionen fuer das Listenbild */ div.gOvItem a img.rFloat, div.gOvItem img.rFloat { float: left; margin: 10px 10px 10px 0; } /* Einstellungen des Listen "mehr..." Links */ div.gOvItem div.more { clear: both; margin-top: 5px; margin-bottom: 5px; } /* Einstellungen des Listen "mehr..." Icons */ div.gOvItem div.more img { display: none; } /*** Layouts ****/ /* layout three columns with image */ table.genOvVar1 { width: 100%; } table.genOvVar1 td { background: #edc352; } table.genOvVar1 td.vSpc, table.genOvVar1 td.hSpc { background: transparent; } table.genOvVar1 td.vSpc img, table.genOvVar1 td.hSpc img { height: 3px; width: 3px; } table.genOvVar1 td.title { vertical-align: top; } table.genOvVar1 td.title h3 { font-size: 13px; margin-top: 0; margin-bottom: 8px; } table.genOvVar1 td.title h3 a { } table.genOvVar1 td.descShell { vertical-align: top; } table.genOvVar1 td.descShell2 { vertical-align: top; white-space: nowrap; } table.genOvVar1 td.title, table.genOvVar1 td.descShell, table.genOvVar1 td.descShell2 { padding: 5px; } table.genOvVar1 td.descShell2 { text-align: center; } /* << */ /* >> Anpassungen des Logos und Bilder im Kopfbereich */ /* Einstellungen der positionierung des Logo Klickbereich */ div.linkCont1 { width: auto; position: absolute; left: 270px; top: 0; z-index: 2; } div.linkCont2 { width: auto; position: absolute; left: 651px; top: 21px; z-index: 3; display: none; } /* Definition der groesse des Logo Klickbereichs*/ div.linkCont1 img { width: 430px; height: 80px; border: 0; } div.linkCont2 img { width: 153px; height: 55px; border: 0; } /* Tabelle fuer Kopfbilder */ table.headerImg { } /* Verhalten des 1 Kopfbereichsbild */ td.hImg { width: 100%; background: url(../../upload/topBg_4270.jpg) top left no-repeat; height: 87px; } /* Verhalten des 2 Kopfbereichsbild */ td.hImg1 { width: 100%; display: none; } /* Verhalten des 3 Kopfbereichsbild */ td.hImg2 { width: 151px; display: none; } /* << */ /* >> Sprachumschalter*/ /* Definitionen fuer den Sprachumschalterkontainer */ div.lSwCont { display: none; } /* Layout fuer Select Box */ div.lSwCont div.selectBox { width: 165px; position: absolute; left: 8px; top: 1px; z-index: 5; } /* Sprachumschalter Formularfeld einstellungen */ div.lSwCont select { font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; font-size: 10px; width: 100%; } /* Layout fuer Flaggen */ div.lSwCont div.flags { width: auto; position: absolute; left: 8px; top: 1px; z-index: 5; font-size: 10px; } div.lSwCont div.flags div.active, div.lSwCont div.flags div.passive { } div.lSwCont div.flags div.language, div.lSwCont div.flags div.flag, div.lSwCont div.flags div.spc { float: left; } div.lSwCont div.flags div.spc, div.lSwCont div.flags div.spc img { display: none; } div.lSwCont div.flags div.language { padding-top: 4px; padding-right: 5px; } div.lSwCont div.flags div.language a { } div.lSwCont div.flags div.flag { padding-right: 10px; } /* << */ /* >>Navigation links*/ /* Aeusserer Navigationskontainer */ div.navOuterShell { } /* Bild fue den Start der Navigation */ div.navOuterShell div.topImg { } div.navOuterShell div.topImg img { display: none; } /* Bild fuer das Ende der Navigation */ div.navOuterShell div.bottomImg { } div.navOuterShell div.bottomImg img { display: none; } /* Anpassung der minimalen Navigationskontainerhoehe */ div.navOuterShell img.minHeight { display: block; height: 1px; float: left; } /*Angaben zur Navigationshuelle. Je Ebene haben Sie die Moeglichkeit den Ebenenhintergrund zu veraendern.*/ div.navCHS_0, div.navCHS_1, div.navCHS_2, div.navCHS_3 { } div.navCHS_1 { border-bottom: 7px solid #E3DEC4; } /* Angaben zur Navigationsknotenhuelle */ div.navNS_0, div.navNS_1, div.navNS_2, div.navNS_3 { text-align: right; } /* Benutzen Sie folgende Klassen um die aeussere Huelle der Navigationslinks zu veraendern.*/ div.navEl_0_normal, div.navEl_0_inPath, div.navEl_0_here, div.navEl_1_normal, div.navEl_1_inPath, div.navEl_1_here, div.navEl_2_normal, div.navEl_2_inPath, div.navEl_2_here, div.navEl_3_normal, div.navEl_3_inPath, div.navEl_3_here { } div.navEl_1_normal, div.navEl_1_inPath, div.navEl_1_here, div.navEl_2_normal, div.navEl_2_inPath, div.navEl_2_here, div.navEl_3_normal, div.navEl_3_inPath, div.navEl_3_here { border-right: 1px solid #E3DEC4; } div.navEl_1_inPath, div.navEl_1_here, div.navEl_2_normal, div.navEl_2_inPath, div.navEl_2_here, div.navEl_3_normal, div.navEl_3_inPath, div.navEl_3_here, div.navEl_4_normal, div.navEl_4_inPath, div.navEl_4_here { border-right: 7px solid #9A2534; } div.navEl_1_normal, div.navEl_1_inPath, div.navEl_1_here { border-top: 1px solid #E3DEC4; } /* Um Bullets vor die Navigationslinks zu Platzieren benutzen Sie die folgenden Klassen*/ div.navEl_0_normal div.outer, div.navEl_0_inPath div.outer, div.navEl_0_here div.outer, div.navEl_1_normal div.outer, div.navEl_1_inPath div.outer, div.navEl_1_here div.outer, div.navEl_2_normal div.outer, div.navEl_2_inPath div.outer, div.navEl_2_here div.outer, div.navEl_3_normal div.outer, div.navEl_3_inPath div.outer, div.navEl_3_here div.outer { } div.navEl_0_normal div.outer, div.navEl_0_inPath div.outer, div.navEl_0_here div.outer { display: none; } div.navEl_2_normal div.outer, div.navEl_2_inPath div.outer, div.navEl_2_here div.outer { border-top: 1px solid #fff; } div.navEl_3_normal div.outer, div.navEl_3_inPath div.outer, div.navEl_3_here div.outer { border-top: 1px solid #E3DEC4; } div.navEl_0_normal span.inner, div.navEl_0_inPath span.inner, div.navEl_0_here span.inner, div.navEl_1_normal span.inner, div.navEl_1_inPath span.inner, div.navEl_1_here span.inner, div.navEl_2_normal span.inner, div.navEl_2_inPath span.inner, div.navEl_2_here span.inner, div.navEl_3_normal span.inner, div.navEl_3_inPath span.inner, div.navEl_3_here span.inner { } /* Ebene 0 */ div.navEl_0_here div.noLink, div.navEl_0_normal a, div.navEl_0_inPath a { display: block; font-weight: bold; font-size: 12px; color: #339; } div.navEl_0_normal a:hover, div.navEl_0_inPath a:hover, div.navEl_0_here div.noLink, div.navEl_0_inPath a { color: #000; } /* Ebene 1 */ div.navEl_1_here div.noLink, div.navEl_1_normal a, div.navEl_1_inPath a { display: block; font-weight: bold; font-size: 12px; color: #5C4714; padding: 10px 20px 10px 0; } div.navEl_1_normal a:hover, div.navEl_1_inPath a div.navEl_1_here div.noLink { color: #9A2534; } div.navEl_1_here div.noLink, div.navEl_1_inPath a { padding-right: 14px; } /* Ebene 2 */ div.navEl_2_here div.noLink, div.navEl_2_normal a, div.navEl_2_inPath a { display: block; font-size: 11px; color: #000; font-weight: bold; padding: 4px 0 4px 0; background: #E3DEC4; } div.navEl_2_normal a:hover, div.navEl_2_here div.noLink, div.navEl_2_inPath a { color: #fff; background: #9A2534; } div.navEl_2_here span.inner, div.navEl_2_normal span.inner, div.navEl_2_inPath span.inner { padding-right: 14px; } /* Ebene 3 */ div.navEl_3_here div.noLink, div.navEl_3_normal a, div.navEl_3_inPath a { display: block; font-size: 11px; color: #000; font-weight: bold; padding: 4px 14px 4px 0; background: #fff; } div.navEl_3_normal a:hover, div.navEl_3_here div.noLink, div.navEl_3_inPath a { color: #9A2534; } /* Ebene 4 */ div.navEl_4_here div.noLink, div.navEl_4_normal a, div.navEl_4_inPath a { display: block; font-size: 11px; color: #000; font-weight: normal; padding: 2px 14px 2px 0; background: #fff; } div.navEl_4_normal a:hover, div.navEl_4_here div.noLink, div.navEl_4_inPath a { color: #9A2534; font-weight: bold; } /* Navigationsbild Kontainer */ div.navigationImage { text-align: center; } /* << */ /* Navigationsbilder */ /* A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */ /* Aktionen */ div.co_aktionen div.navEl_1_here div.noLink, div.co_aktionen div.navEl_1_normal a, div.co_aktionen div.navEl_1_inPath a { background: url(../../upload/aktionen_3992.gif) 10px center no-repeat; } div.co_aktionen div.navEl_1_normal a:hover, div.co_aktionen div.navEl_1_here div.noLink, div.co_aktionen div.navEl_1_inPath a { background-image: url(../../upload/aktionen2_3993.gif); } /* ASA */ div.co_asa div.navEl_1_here div.noLink, div.co_asa div.navEl_1_normal a, div.co_asa div.navEl_1_inPath a { background: url(../../upload/asa_2772.gif) 10px center no-repeat; } /* Blomus */ div.co_blomus div.navEl_1_here div.noLink, div.co_blomus div.navEl_1_normal a, div.co_blomus div.navEl_1_inPath a { background: url(../../upload/blomus_2773.gif) 10px center no-repeat; } /* Brainstream */ div.co_brainstream div.navEl_1_here div.noLink, div.co_brainstream div.navEl_1_normal a, div.co_brainstream div.navEl_1_inPath a { background: url(../../upload/brainstream_2745.gif) 10px center no-repeat; } div.co_brainstream div.navEl_1_normal a:hover, div.co_brainstream div.navEl_1_here div.noLink, div.co_brainstream div.navEl_1_inPath a { background-image: url(../../upload/brainstream2_2746.gif); } /* Bodypaint */ div.co_bodypaint div.navEl_1_here div.noLink, div.co_bodypaint div.navEl_1_normal a, div.co_bodypaint div.navEl_1_inPath a { background: url(../../upload/bodypaint_2807.gif) 10px center no-repeat; } div.co_bodypaint div.navEl_1_normal a:hover, div.co_bodypaint div.navEl_1_here div.noLink, div.co_bodypaint div.navEl_1_inPath a { background-image: url(../../upload/bodypaint2_2808.gif); } /* Colony */ div.co_colony div.navEl_1_here div.noLink, div.co_colony div.navEl_1_normal a, div.co_colony div.navEl_1_inPath a { background: url(../../upload/colony_2774.gif) 10px center no-repeat; } /* Conmoto */ div.co_conmoto div.navEl_1_here div.noLink, div.co_conmoto div.navEl_1_normal a, div.co_conmoto div.navEl_1_inPath a { background: url(../../upload/conmoto_2749.gif) 10px center no-repeat; } div.co_conmoto div.navEl_1_normal a:hover, div.co_conmoto div.navEl_1_here div.noLink, div.co_conmoto div.navEl_1_inPath a { background-image: url(../../upload/conmoto2_2750.gif); } /* Delux */ div.co_delux div.navEl_1_here div.noLink, div.co_delux div.navEl_1_normal a, div.co_delux div.navEl_1_inPath a { background: url(../../upload/deluxe2_6445.gif) 10px center no-repeat; } div.co_delux div.navEl_1_normal a:hover, div.co_delux div.navEl_1_here div.noLink, div.co_delux div.navEl_1_inPath a { background-image: url(../../upload/deluxe_6444.gif); } /* El Casco */ div.co_elcasco div.navEl_1_here div.noLink, div.co_elcasco div.navEl_1_normal a, div.co_elcasco div.navEl_1_inPath a { background: url(../../upload/elcasco_2775.gif) 10px center no-repeat; } /* Eschenbach */ div.co_eschenbach div.navEl_1_here div.noLink, div.co_eschenbach div.navEl_1_normal a, div.co_eschenbach div.navEl_1_inPath a { background: url(../../upload/eschenbach2_7983.gif) 10px center no-repeat;} div.co_eschenbach div.navEl_1_normal a:hover, div.co_eschenbach div.navEl_1_here div.noLink, div.co_eschenbach div.navEl_1_inPath a { background-image: url(../../upload/eschenbach_7982.gif); } /* Herrnhuter Sterne */ div.co_hernhuter div.navEl_1_here div.noLink, div.co_hernhuter div.navEl_1_normal a, div.co_hernhuter div.navEl_1_inPath a { background: url(../../upload/hernhuter_2794.gif) 10px center no-repeat; } div.co_hernhuter div.navEl_1_normal a:hover, div.co_hernhuter div.navEl_1_here div.noLink, div.co_hernhuter div.navEl_1_inPath a { background-image: url(../../upload/hernhuter2_2795.gif); } /* Hussel */ div.co_hussel div.navEl_1_here div.noLink, div.co_hussel div.navEl_1_normal a, div.co_hussel div.navEl_1_inPath a { background: url(../../upload/hussel_2753.gif) 10px center no-repeat; } div.co_hussel div.navEl_1_normal a:hover, div.co_hussel div.navEl_1_here div.noLink, div.co_hussel div.navEl_1_inPath a { background-image: url(../../upload/hussel2_2754.gif); } /* Kneisz Design */ div.co_kneisz div.navEl_1_here div.noLink, div.co_kneisz div.navEl_1_normal a, div.co_kneisz div.navEl_1_inPath a { background: url(../../upload/kneisz_2813.gif) 10px center no-repeat; } div.co_kneisz div.navEl_1_normal a:hover, div.co_kneisz div.navEl_1_here div.noLink, div.co_kneisz div.navEl_1_inPath a { background-image: url(../../upload/kneisz2_2814.gif); } /* Kahla */ div.co_kahla div.navEl_1_here div.noLink, div.co_kahla div.navEl_1_normal a, div.co_kahla div.navEl_1_inPath a { background: url(../../upload/kahla_2776.gif) 10px center no-repeat; } /* Keilbach */ div.co_keilbach div.navEl_1_here div.noLink, div.co_keilbach div.navEl_1_normal a, div.co_keilbach div.navEl_1_inPath a { background: url(../../upload/keilbach_2777.gif) 10px center no-repeat; } div.co_keilbach div.navEl_1_normal a:hover, div.co_keilbach div.navEl_1_here div.noLink, div.co_keilbach div.navEl_1_inPath a { background-image: url(../../upload/keilbach2_2759.gif); } /* Lardon */ div.co_lardon div.navEl_1_here div.noLink, div.co_lardon div.navEl_1_normal a, div.co_lardon div.navEl_1_inPath a { background: url(../../upload/lardon_sw_9042.gif) 10px center no-repeat; } div.co_lardon div.navEl_1_normal a:hover, div.co_lardon div.navEl_1_here div.noLink, div.co_lardon div.navEl_1_inPath a { background-image: url(../../upload/lardon_9041.gif); } /* LaNature */ div.co_lanature div.navEl_1_here div.noLink, div.co_lanature div.navEl_1_normal a, div.co_lanature div.navEl_1_inPath a { background: url(../../upload/lanature_2760.gif) 10px center no-repeat; } div.co_lanature div.navEl_1_normal a:hover, div.co_lanature div.navEl_1_here div.noLink, div.co_lanature div.navEl_1_inPath a { background-image: url(../../upload/lanature2_2761.gif); } /* Meybona */ div.co_meybona div.navEl_1_here div.noLink, div.co_meybona div.navEl_1_normal a, div.co_meybona div.navEl_1_inPath a { background: url(../../upload/meybona_2800.gif) 10px center no-repeat; } div.co_meybona div.navEl_1_normal a:hover, div.co_meybona div.navEl_1_here div.noLink, div.co_meybona div.navEl_1_inPath a { background-image: url(../../upload/meybona2_2801.gif); } /* MBM */ div.co_mbm div.navEl_1_here div.noLink, div.co_mbm div.navEl_1_normal a, div.co_mbm div.navEl_1_inPath a { background: url(../../upload/mbm_2778.gif) 10px center no-repeat; } /* menue */ div.co_menu div.navEl_1_here div.noLink, div.co_menu div.navEl_1_normal a, div.co_menu div.navEl_1_inPath a { background: url(../../upload/menu_2779.gif) 10px center no-repeat; } /* Oralfixation */ div.co_oralfixation div.navEl_1_here div.noLink, div.co_oralfixation div.navEl_1_normal a, div.co_oralfixation div.navEl_1_inPath a { background: url(../../upload/oralfixation_2780.gif) 10px center no-repeat; } /* Philippi */ div.co_philippi div.navEl_1_here div.noLink, div.co_philippi div.navEl_1_normal a, div.co_philippi div.navEl_1_inPath a { background: url(../../upload/philippi_2781.gif) 10px center no-repeat; } /* PatSaysNow */ div.co_patsaysnow div.navEl_1_here div.noLink, div.co_patsaysnow div.navEl_1_normal a, div.co_patsaysnow div.navEl_1_inPath a { background: url(../../upload/patsaysnow_2770.gif) 10px center no-repeat; } div.co_patsaysnow div.navEl_1_normal a:hover, div.co_patsaysnow div.navEl_1_here div.noLink, div.co_patsaysnow div.navEl_1_inPath a { background-image: url(../../upload/patsaysnow2_2771.gif); } /* Raeder Geschenke */ div.co_raeder div.navEl_1_here div.noLink, div.co_raeder div.navEl_1_normal a, div.co_raeder div.navEl_1_inPath a { background: url(../../upload/raeder_2782.gif) 10px center no-repeat; } div.co_raeder div.navEl_1_normal a:hover, div.co_raeder div.navEl_1_here div.noLink, div.co_raeder div.navEl_1_inPath a { background-image: url(../../upload/raeder2_2783.gif); } /* Sitting Bull */ div.co_sittingbull div.navEl_1_here div.noLink, div.co_sittingbull div.navEl_1_normal a, div.co_sittingbull div.navEl_1_inPath a { background: url(../../upload/sittingbull_sw_6802.gif) 10px center no-repeat; } div.co_sittingbull div.navEl_1_normal a:hover, div.co_sittingbull div.navEl_1_here div.noLink, div.co_sittingbull div.navEl_1_inPath a { background-image: url(../../upload/sittingbull_6801.gif); } /* Reed Diffuser */ div.co_reeddiffuser div.navEl_1_here div.noLink, div.co_reeddiffuser div.navEl_1_normal a, div.co_reeddiffuser div.navEl_1_inPath a { } div.co_reeddiffuser div.navEl_1_normal a:hover, div.co_reeddiffuser div.navEl_1_here div.noLink, div.co_reeddiffuser div.navEl_1_inPath a { } /* Roensch */ div.co_roensch div.navEl_1_here div.noLink, div.co_roensch div.navEl_1_normal a, div.co_roensch div.navEl_1_inPath a { background: url(../../upload/roensch_7981.gif) 10px center no-repeat; } div.co_roensch div.navEl_1_normal a:hover, div.co_roensch div.navEl_1_here div.noLink, div.co_roensch div.navEl_1_inPath a { background-image: url(../../upload/roensch2_2811.gif); } /* Sheepworld */ div.co_sheepworld div.navEl_1_here div.noLink, div.co_sheepworld div.navEl_1_normal a, div.co_sheepworld div.navEl_1_inPath a { background: url(../../upload/sheepworld_2789.gif) 10px center no-repeat; } /* Skia */ div.co_skia div.navEl_1_here div.noLink, div.co_skia div.navEl_1_normal a, div.co_skia div.navEl_1_inPath a { background: url(../../upload/skia_2790.gif) 10px center no-repeat; } div.co_skia div.navEl_1_normal a:hover, div.co_skia div.navEl_1_here div.noLink, div.co_skia div.navEl_1_inPath a { background-image: url(../../upload/skia2_2791.gif); } /* Viteo */ div.co_viteo div.navEl_1_here div.noLink, div.co_viteo div.navEl_1_normal a, div.co_viteo div.navEl_1_inPath a { background: url(../../upload/viteo_3941.gif) 10px center no-repeat; } div.co_viteo div.navEl_1_normal a:hover, div.co_viteo div.navEl_1_here div.noLink, div.co_viteo div.navEl_1_inPath a { background-image: url(../../upload/viteo2_3942.gif); } /* Zotter */ div.co_zotter div.navEl_1_here div.noLink, div.co_zotter div.navEl_1_normal a, div.co_zotter div.navEl_1_inPath a { background: url(../../upload/zotter_2805.gif) 10px center no-repeat; } div.co_zotter div.navEl_1_normal a:hover, div.co_zotter div.navEl_1_here div.noLink, div.co_zotter div.navEl_1_inPath a { background-image: url(../../upload/zotter2_2804.gif); } /* << */ /* >> Teaser Navigation */ /* Servicecenter */ /* Titel */ div.co_servicecenter div.nssTitle div.tDes1 h3 { font-size: 14px; margin: 0; color: #fff; display: block; background: url(../xist4c/web/krusekopf/01/img/teaserheadbg.gif) top left repeat-x; padding: 4px 7px 4px 7px; } /* Aeussere Navigationsheulle */ div.co_servicecenter div.nssDes1 { background: url(../xist4c/web/krusekopf/01/img/teaserbg.gif) top left repeat; } div.co_servicecenter div.nssDes2 { background: url(../xist4c/web/krusekopf/01/img/teaserbgtop.gif) top left repeat-x; } /*Angaben zur Navigationshuelle. Je Ebene haben Sie die Moeglichkeit den Ebenenhintergrund zu veraendern.*/ div.co_servicecenter div.navCHS_1, div.co_servicecenter div.navCHS_2, div.co_servicecenter div.navCHS_3 { border: 0; } /* Angaben zur Navigationsknotenhuelle */ div.co_servicecenter div.navNS_0, div.co_servicecenter div.navNS_1, div.co_servicecenter div.navNS_2, div.co_servicecenter div.navNS_3 { text-align: left; } /* Benutzen Sie folgende Klassen um die aeussere Huelle der Navigationslinks zu veraendern.*/ div.co_servicecenter div.navEl_1_normal, div.co_servicecenter div.navEl_1_inPath, div.co_servicecenter div.navEl_1_here, div.co_servicecenter div.navEl_2_normal, div.co_servicecenter div.navEl_2_inPath, div.co_servicecenter div.navEl_2_here, div.co_servicecenter div.navEl_3_normal, div.co_servicecenter div.navEl_3_inPath, div.co_servicecenter div.navEl_3_here { border: 0; } div.rElCont div.co_servicecenter div.navEl_1_normal, div.co_servicecenter div.navEl_1_inPath, div.co_servicecenter div.navEl_1_here { border-bottom: 1px solid #fff; } /* Ebene 1 */ div.co_servicecenter div.navEl_1_here div.noLink, div.co_servicecenter div.navEl_1_normal a, div.co_servicecenter div.navEl_1_inPath a { display: block; font-size: 11px; color: #000; padding: 4px 5px 4px 7px; } div.co_servicecenter div.navEl_1_normal a:hover, div.co_servicecenter div.navEl_1_inPath a:hover, div.co_servicecenter div.navEl_1_here div.noLink, div.co_servicecenter div.navEl_1_inPath a { color: #6E6D26; } /* Infotek */ div.co_infotek { margin-bottom: 20px; } /* << */ /* >> Linkliste */ /* Linkliste Tabelle */ table.llShell { margin: 0 15px 20px 15px; } /* Einstellungen fuer den Linklisten Titel fuer normal, besucht und bei ueberfahren mit der Maus */ table.llShell td a { font-size: 13px; color: #f90; font-weight: bold; } table.llShell td a:hover { color: #000; } /* Linklisten Bullet */ table.llShell td img.bullet { width: 6px; height: 6px; margin-top: 5px; margin-right: 6px; background: #574313; } /* Anpassungen der Linkbeschreibung */ table.llShell td.desc { font-size: 11px; } /* << */ /* >> Login und Logout Element */ /* >>>>>Login */ /* Definitionen fuer den Schnelllogin- Schnelllogoutkontainer */ div.quickLogKontainer { width: 165px; position: absolute; left: 823px; top: 82px; z-index: 4; display: none; } /* Einstellungen fuer das Login und Logout Formular */ form.quicklogin, form.login, form.logoutFormField { margin: 0; padding: 0; } /* Angaben zu den Input Formularfeldern im Quicklogin Bereich*/ form.quicklogin table td input.text { font-size: 10px; width: 100%; } /* Angaben zu dem Input Button im Quicklogin Bereich*/ form.quicklogin table td input.button { font-weight: auto; } /* Beschriftungen fuer den Standard Login */ form.login table td span { } /* Angaben zu den Input Formularfeldern im Standard Login Bereich*/ form.login table td input.text { font-size: 10px; width: 160px; } /* Angaben zu dem Input Button im Standard Login*/ form.login table td input.button { cursor: pointer; /cursor: hand; width: auto; } /* Einstellungen zur Standard Login Tabelle */ table.designShell { border: 1px solid #5C4714; background: #E3DEC4; } /* Anpassen der Login Fehlermeldung bei missgluecktem Loginversuch*/ form.login table.designShell td table td div { font-weight: bold; color: #9A2534; } /* >>>>Logout */ /* logout Tabelle */ table.logout { } /* Anpassungen fuer den Beschreibungstext im Logout Bereich */ table.logout td span { font-size: 9px; font-weight: normal; } /* Formatierung des Benutzernamens im Logout Bereich.*/ table.logout td { font-weight: bold; font-size: 10px; white-space: nowrap; line-height: 10px; } /* Formatierung des Buttons im Logout Bereich.*/ table.logout td a img { } /* >> */ /* >> News Uebersicht*/ /* News Uebersicht Tabelle */ table.news { } /* Anpassungen fuer das News Datum in der Uebersicht*/ table.news td.date { font-weight: bold; color: #666; font-size: 11px; padding-right: 8px; } /* News Info Spalte */ table.news td.item { padding-bottom: 12px; } /* Definition der News Uebersicht Ueberschrift */ table.news td.item h3 { margin: 0 0 3px 0; } /* Definition der News Uebersicht Unterueberschrift */ table.news td.item h4 { margin: 0 0 3px 0; } /* Einstellungen fuer die News Beschreibung */ table.news td.item div.desc { margin: 0; } /* Definition fuer das News Bild */ table.news td.item img.rFloat { float: right; margin-left: 10px; margin-bottom: 5px; } /* Anpassungen des News "mehr...." Links */ table.news td.item div.more { clear: both; margin-top: 5px; margin-bottom: 5px; text-align: left; font-size: 11px; } /* Einstellungen des News "mehr..." Icons */ table.news td.item div.more img { } /* << */ /* >> News Teaser */ div.newsTContShell { padding-bottom: 10px; } div.newsTContShell div.inner { } div.newsTContShell div.inner div.date { font-weight: bold; color: #000; font-size: 10px; } div.newsTContShell div.inner h3 { font-size: 11px; margin: 0; font-weight: normal; margin-bottom: 5px; padding-left: 10px; } div.newsTContShell div.inner h3 a { } /* << */ /* >> Blaetterelement */ /* Pager Kontainer */ div.pager { margin-bottom: 3px; } /* Einstellungen fuer die Informationen wieviel Seiten gefunden wurden bsp. "Seite (1 / 23)" */ div.pager table.pInfo td, div.pager table.pInfo td span { font-size: 11px; color: #333; display: none; } /* Definition der Schriftfarbe der Tabellenzelle fuer die gefundenen Seiten */ div.pager table.pPages td { color: #369; } /* Einstellung der momentan angewaehlten Seite */ div.pager table.pPages td span { font-weight: bold; color: #fff; display: block; background: #9A2534; height: 16px; width: 20px; padding: 0 5px; } /* Anpassen der Links */ div.pager table.pPages td a { font-weight: bold; vertical-align: middle; color: #000; display: block; height: 16px; width: 20px; padding: 0 5px; } div.pager table.pPages td a:hover { background: #9A2534; color: #fff; } /* Anpassungen fuer den linken Pager Pfeil aktiv und passiv */ div.pager table.pPages td a img.firstAct, div.pager table.pPages td img.firstPass { height: 16px; width: 20px; background: url(../xist4c/web/krusekopf/01/img/firstPageButtActive.gif) center center no-repeat; margin: 0 4px 0 0; border: 0; } div.pager table.pPages td a:hover img.firstAct, div.pager table.pPages td img.firstPass { background-image: url(../xist4c/web/krusekopf/01/img/firstPageButtPassive.gif); } /* Anpassungen fuer den halb linken Pager Pfeil aktiv und passiv */ div.pager table.pPages td a img.prevAct, div.pager table.pPages td img.prevPass { height: 16px; width: 20px; margin: 0 2px 0 0; background: url(../xist4c/web/krusekopf/01/img/prevPageButtActive.gif) center center no-repeat; border: 0; } div.pager table.pPages td a:hover img.prevAct, div.pager table.pPages td img.prevPass { background-image: url(../xist4c/web/krusekopf/01/img/prevPageButtPassive.gif); } /* Anpassungen fuer den halb rechten Pager Pfeil aktiv und passiv */ div.pager table.pPages td a img.nextAct, div.pager table.pPages td img.nextPass { height: 16px; width: 20px; margin: 0 0 0 2px; background: url(../xist4c/web/krusekopf/01/img/nextPageButtActive.gif) center center no-repeat; border: 0; } div.pager table.pPages td a:hover img.nextAct, div.pager table.pPages td img.nextPass { background-image: url(../xist4c/web/krusekopf/01/img/nextPageButtPassive.gif); } /* Anpassungen fuer den rechten Pager Pfeil aktiv und passiv */ div.pager table.pPages td a img.lastAct, div.pager table.pPages td img.lastPass { height: 16px; width: 20px; margin: 0 0 0 4px; background: url(../xist4c/web/krusekopf/01/img/lastPageButtActive.gif) center center no-repeat; border: 0; } div.pager table.pPages td a:hover img.lastAct, div.pager table.pPages td img.lastPass { background: url(../xist4c/web/krusekopf/01/img/lastPageButtPassive.gif) center center no-repeat; } /* Anpassungen fuer das Pager Trennelement */ div.pager table.pPages td img.sep { height: 16px; width: 1px; border: 0; background: #E3DEC4; } /* << */ /* >> Blaetterelement fuer Dokumentunterteilung */ div.topContentPager { } div.bottomContentPager { } div.contPagerShell { background: #efefef; margin-bottom: 10px; } div.contPagerShell div.outDes1 { padding: 2px 5px 2px 5px; border: 1px solid #b3b3b3; } div.contPagerShell div.outDes2 { text-align: right; } div.contPagerShell div.outDes2 table { margin-left: auto; } div.contPagerShell div.outDes2 table td { } div.contPagerShell div.outDes2 table td div.des1 { } div.contPagerShell div.outDes2 table td div.des2 { } /* Links des normalen pagers */ div.contPagerShell div.outDes2 table td div.here, div.contPagerShell div.outDes2 table td a { display: block; font-size: 12px; line-height: 12px; font-weight: bold; padding: 1px 4px 1px 4px; border: 1px solid #000; color: #fff; background: #a0b8cf; } div.contPagerShell div.outDes2 table td a:hover { background: #7ca4c7; } div.contPagerShell div.outDes2 table td div.here { background: #5589b7; } div.contPagerShell div.outDes2 table td div.spc { display: block; width: 1px; height: 10px; margin: 1px 3px 0 3px; background: #000; } /* Anordnung der simplen Bildpagerelemente */ div.smpContPagerShell { background: #efefef; margin-bottom: 10px; } * html div.smpContPagerShell { width: 100%; } div.smpContPagerShell div.outDes1 { border: 1px solid #b3b3b3; } div.smpContPagerShell div.outDes2 { margin: 2px 5px 2px 5px; text-align: right; } * html div.smpContPagerShell div.outDes2 { width: 100%; } div.smpContPagerShell div.outDes2 table.outer { margin-left: auto; } div.smpContPagerShell div.outDes2 table.outer td.left { text-align: left; } div.smpContPagerShell div.outDes2 table.outer td.right { text-align: right; } div.smpContPagerShell div.outDes2 table.outer td.left div.d1_left, div.smpContPagerShell div.outDes2 table.outer td.right div.d1_right { } div.smpContPagerShell div.outDes2 table.outer td.left div.d2_left, div.smpContPagerShell div.outDes2 table.outer td.right div.d2_right { } div.smpContPagerShell div.outDes2 table.outer td.sep { width: 1px; } table.pElOuter_left, table.pElOuter_right { } table.pElOuter_right { margin-left: auto; } table.pElOuter_left td.spcL, table.pElOuter_left td.spcR, table.pElOuter_right td.spcL, table.pElOuter_right td.spcR { } table.pElOuter_left td.spcR img, table.pElOuter_right td.spcL img { width: 3px; } table.pElOuter_left td.spcL img, table.pElOuter_left td.spcR img, table.pElOuter_right td.spcL img, table.pElOuter_right td.spcR img { } /* Links des simplen Bildpagerelements */ table.pElOuter_left td.pEl div.noLink img, table.pElOuter_right td.pEl div.noLink img, table.pElOuter_left td.pEl a img, table.pElOuter_right td.pEl a img { height: 20px; width: 24px; background-position: left top; background-repeat: no-repeat; border: none; } table.pElOuter_left td.pEl a img { background-image: url(../xist4c/web/krusekopf/01/img/smpPagArrowLeft.gif); } table.pElOuter_right td.pEl a img { background-image: url(../xist4c/web/krusekopf/01/img/smpPagArrowRight.gif); } table.pElOuter_left td.pEl div.noLink img { background-image: url(../xist4c/web/krusekopf/01/img/smpPagArrowLeftPass.gif); } table.pElOuter_right td.pEl div.noLink img { background-image: url(../xist4c/web/krusekopf/01/img/smpPagArrowRightPass.gif); } /* Anordnung der simplen Textpagerelemente */ div.smpTxtContPagerShell { background: #efefef; margin-bottom: 10px; } * html div.smpTxtContPagerShell { width: 100%; } div.smpTxtContPagerShell div.outDes1 { border: 1px solid #b3b3b3; padding: 2px 5px 2px 5px; } div.smpTxtContPagerShell div.outDes2 { text-align: right; } * html div.smpTxtContPagerShell div.outDes2 { width: 100%; } div.smpTxtContPagerShell div.outDes2 table.outer { margin-left: auto; } div.smpTxtContPagerShell div.outDes2 table.outer td.left { text-align: left; } div.smpTxtContPagerShell div.outDes2 table.outer td.right { text-align: right; } div.smpTxtContPagerShell div.outDes2 table.outer td.left div.d1_left, div.smpTxtContPagerShell div.outDes2 table.outer td.right div.d1_right { } div.smpTxtContPagerShell div.outDes2 table.outer td.left div.d2_left, div.smpTxtContPagerShell div.outDes2 table.outer td.right div.d2_right { } div.smpTxtContPagerShell div.outDes2 table.outer td.sep { width: 1px; background: #000; } table.pTxtElOuter_left, table.pTxtElOuter_right { } table.pTxtElOuter_right { margin-left: auto; } table.pTxtElOuter_left td.spcL, table.pTxtElOuter_left td.spcR, table.pTxtElOuter_right td.spcL, table.pTxtElOuter_right td.spcR { } table.pTxtElOuter_left td.spcR img, table.pTxtElOuter_right td.spcL img { width: 10px; } table.pTxtElOuter_left td.spcL img, table.pTxtElOuter_left td.spcR img, table.pTxtElOuter_right td.spcL img, table.pTxtElOuter_right td.spcR img { } /* Links des simplen Textpagerelements */ table.pTxtElOuter_left td.pEl div.noLink, table.pTxtElOuter_right td.pEl div.noLink, table.pTxtElOuter_left td.pEl a, table.pTxtElOuter_right td.pEl a { display: block; font-size: 12px; border: none; } table.pTxtElOuter_left td.pEl div.noLink, table.pTxtElOuter_left td.pEl a { padding-left: 20px; background: url(../xist4c/web/krusekopf/01/img/smpTxtPagArrowLeft.gif) 0 1px no-repeat; } table.pTxtElOuter_right td.pEl div.noLink, table.pTxtElOuter_right td.pEl a { padding-right: 20px; background: url(../xist4c/web/krusekopf/01/img/smpTxtPagArrowRight.gif) right 1px no-repeat; } table.pTxtElOuter_left td.pEl div.noLink { background-image: url(../xist4c/web/krusekopf/01/img/smpTxtPagArrowLeftPass.gif); color: #666; } table.pTxtElOuter_right td.pEl div.noLink { background-image: url(../xist4c/web/krusekopf/01/img/smpTxtPagArrowRightPass.gif); color: #666; } /* << */ /* >> Absaetze im Panel Modus */ /* Panel Titelkontainer */ div.panelTitle, table.footer td div.panelTitle { padding: 4px 15px 4px 15px; margin-bottom: 1px; } * html div.panelTitle, * html table.foot er td div.panelTitle { width: 100%; } /* Panel Titel */ div.panelTitle h3, table.footer td div.panelTitle h3 { font-size: 14px; margin: 0; color: #574313; } /* Enstellungen fuer den Panelkontainer */ div.panelOuter, table.footer td div.panelOuter { margin-bottom: 12px; } * html div.panelOuter, * html table.footer td div.panelOuter { width: 100%; } /* Einstellungen fuer den Inhaltskontainer */ div.panelOuter div.desOut1 div.des1 { } div.panelOuter div.desOut1 div.des2 { } div.panelOuter div.desOut1 div.des2 div.cont, table.footer td div.panelOuter div.desOut1 div.des2 div.cont { padding: 5px 15px; } * html div.panelOuter div.desOut1 div.des2 div.cont, * html table.footer td div.panelOuter div.desOut1 div.des2 div.cont { width: 100%; } div.panelOuter p, table.footer td div.panelOuter p { color: #574313; } div.panelOuter a, table.footer td div.panelOuter a { color: #000; } div.panelOuter a:hover, table.footer td div.panelOuter a:hover { color: #f90; } /* Design Panel Titelkontainer */ div.designTitle, div.co_orange div.designTitle, table.footer td div.designTitle { background: url(../xist4c/web/krusekopf/01/img/paneltitlebg.gif) top left repeat-x; padding: 4px 15px; margin-bottom: 1px; } div.co_orange div.designTitle { background-image: url(../xist4c/web/krusekopf/01/img/despaneltitlebg.gif); } * html div.designTitle, * html table.footer td div.designTitle { width: 100%; } /* Design Panel Titel */ div.designTitle h3, div.co_orange div.designTitle h3, table.footer td div.designTitle h3 { font-size: 14px; margin: 0; color: #574313; } /* Enstellungen fuer den Design Panelkontainer */ div.desPanelOuter, div.co_orange div.desPanelOuter, table.footer td div.desPanelOuter { margin-bottom: 12px; } * html div.desPanelOuter, * html table.footer td div.desPanelOuter { width: 100%; } /* Einstellungen fuer den Inhaltskontainer */ div.desPanelOuter div.desOut1 div.des1, div.co_orange div.desPanelOuter div.desOut1 div.des1 { background: url(../xist4c/web/krusekopf/01/img/panelbg.gif) top left repeat; } div.co_orange div.desPanelOuter div.desOut1 div.des1 { background-image: url(../xist4c/web/krusekopf/01/img/despanelbg.gif); } div.desPanelOuter div.desOut1 div.des2, div.co_orange div.desPanelOuter div.desOut1 div.des2 { background: url(../xist4c/web/krusekopf/01/img/paneltopbg.gif) top left repeat-x; } div.co_orange div.desPanelOuter div.desOut1 div.des2 { background-image: url(../xist4c/web/krusekopf/01/img/despaneltopbg.gif); } /* Einstellungen fuer den Design Inhaltskontainer */ div.desPanelOuter div.desOut1 div.des2 div.cont, div.co_orange div.desPanelOuter div.desOut1 div.des2 div.cont, table.footer td div.desPanelOuter div.desOut1 div.des2 div.cont { padding: 5px 15px; } * html div.desPanelOuter div.desOut1 div.des2 div.cont, * html table.footer td div.desPanelOuter div.desOut1 div.des2 div.cont { width: 100%; } /* Farben */ div.desPanelOuter, div.co_orange div.desPanelOuter { color: #574313; } div.desPanelOuter a, div.co_orange div.desPanelOuter a { color: #000; } div.desPanelOuter a:hover, div.co_orange div.desPanelOuter a:hover { color: #f90; } /* << */ /* >> Absaetze Normal */ /* Absatz Titel */ h3.paraTitle { display: block; font-style: normal; font-weight: bold; padding-top: 6px; margin-bottom: 5px; } /* Absatz Titel im Footerbereich*/ table.footer td h3.paraTitle { display: block; font-style: normal; font-weight: bold; font-size: 11px; padding-top: 3px; margin-bottom: 2px; } /* Einstellungen der Bild und Content ausrichtung */ div.paraImgOuter, div.paraImgOuterL, div.paraImgOuterR, div.paraFxImgOuter { padding-top: 2px; margin-bottom: 3px; } /* Fuer Bild zu Textabstand bei links und rechts umfliessend */ div.paraImgOuterL { padding-right: 6px; } div.paraImgOuterR { padding-left: 6px; } div.paraContOuter, div.paraContOuterL, div.paraContOuterR { } /* Bild zu Textabstand bei rechts und links freibleibend*/ div.paraContOuterL { padding-left: 6px; } div.paraContOuterR { padding-right: 6px; } /* Innerer Bild Container */ div.paraImgInner { margin-left: auto; margin-right: auto; } /* Bild zu Text im zentrierten Modus */ div.paraCeImgOuter { text-align: center; } /* Definition Absatz Untertitel */ h4.paraSubtitle { font-weight: bold; font-size: 13px; margin: 2px 0 5px 0; } /* Definition Absatz Untertitel im Footerbereich*/ table.footer td h4.paraSubtitle { font-weight: bold; font-size: 11px; margin: 2px 0 2px 0; } /* Einstellungen Absatz Inhalt */ table.contentTable td.contentColumn p, table.contentTable td.contentColumn p.paraEl { margin: 0; margin-bottom: 100px; } /* Einstellungen Absatz Inhalt im Footerbereich*/ table.footer td p, table.footer td p.paraEl { margin: 0; margin-bottom: 5px; } /* Definition Bilduntertext */ div.paraImgInner div.sT { margin-top: 2px; margin-bottom: 3px; font-size: 11px; text-align: left; } /* Definition Bilduntertext im Footerbereich */ table.footer td div.paraImgInner div.sT { margin-top: 1px; margin-bottom: 2px; font-size: 10px; text-align: left; } /* << */ /* >> Allgemeine Absatz und Panel Definitionen */ /* Nicht aendern!! */ img.clearAll { display: block; clear: both; visibility: hidden; } /* Standard definition des Absatzabstandes */ div.contentContainer table.contentTable td.contentColumn p { margin: 0; margin-bottom: 10px; } /* Standard definition des Absatzabstandes im Footerbereich*/ table.footer td div.contentContainer table.contentTable td.contentColumn p { margin: 0; margin-bottom: 7px; } /* Einstellungen fuer den Absatz "mehr..." Link */ div.paraSublinkShell { text-align: left; margin-bottom: 5px; font-size: 11px; } div.paraSublinkShell img { padding-left: 5px; } /* Absatz Sublink */ div.paraSublinkShell span.sL { } /* Einstellungen fuer den Absatz "mehr..." Link im Footerbereich */ table.footer td div.paraSublinkShell { } table.footer td div.paraSublinkShell span.sL { } /* Einstellungen der Absatz und Panel "mehr.." Link Icons */ div.paraSublinkShell span.sL img { width: 8px; height: 14px; vertical-align: text-top; border: 0; background: url(../xist4c/web/krusekopf/01/img/paragraphArrow.gif) left 1px no-repeat; } /* Einstellungen der Absatz und Panel "mehr.." Link Icons im Footerbereich */ table.footer td div.paraSublinkShell span.sL img { display: none; } /* << */ /* >> Schnellzugriff*/ /* Definitionen fuer den Schnellzugriffkontainer */ div.qAccessCont { width: 155px; position: absolute; left: 7px; top: 128px; z-index: 6; display: none; } /* Schnellzugriff Formularfeld einstellungen */ div.qAccessCont select { font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; font-size: 10px; width: 100%; } /* << */ /* >> Schnellsuche*/ /* Definitionen fuer den Schnellsuchekontainer */ div.qSearchCont { width: 155px; position: absolute; left: 7px; top: 107px; z-index: 3; display: none; } /* Schnellsuche Formulareinstellungen */ div.qSearchCont form { margin-top: 0; margin-bottom: 0; } div.qSearchCont table.qSearch { } /* Schnellsuche Formularfeld Einstellungen */ div.qSearchCont table.qSearch td input.text { font-size: 10px; width: 100%; } /* Schnellsuche "Go" button */ div.qSearchCont table.qSearch td input.button { } /* << */ /* >> Standard Suche */ /* Standard Suchfeld Einstellungen */ table.stdSearch td input.text { font-size: 13px; width: 300px; } /* Standard Suche Button Einstellungen */ table.stdSearch td input.button { cursor: pointer; /cursor: hand; border: 1px solid #E3DEC4; font-size: 12px; } /* Suche Formulareinstellungen */ table.stdSearch form { margin-top: 0; margin-bottom: 0; } /* Einstellungen fuer die farbige Hinterlegung im Suchergebnis */ span.searchResult { padding: 0 2px; background: #E3DEC4; } /* << */ /* >> Empfehlung versenden */ /* Empfehlung versenden Tabelle */ table.recomShell { } /* Anpassungen des Formulars */ table.recomShell form { padding: 0; margin: 0; } /* Einstellungen fuer die Fehlermeldungen bei unkorrektem Ausfuellen der Formularelemente */ table.recomShell td ul li { font-weight: bold; font-size: 11px; color: darkred; margin: 3px 0 3px 0; } /* Anpassungen der Formular Tabellenspalten */ table.recomShell td form table td { padding: 1px 8px 1px 8px; } /* Anpassungen der Formularelemente */ table.recomShell td form table td input.text, table.recomShell td form table td textarea { font-size: 11px; font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; width: 100%; margin-bottom: 5px; padding-left: 1px; padding-right: 1px; } table.recomShell td form table td textarea { overflow: auto; } /* Einstellungen fuer die Input Formularelemente */ table.recomShell td form table td input.button { font-size: 11px; margin-bottom: 8px; cursor: pointer; /cursor: hand; } /* Angaben zum Kontainer der Sendebestaetigung */ div.recomSuccess { font-weight: bold; font-size: 13px; color: #333; margin: 10px 0; } /* Einstellungen fuer den Kontainer der Nutzungsbedingungen */ table.recomShell td div.policy { font-size: 11px; color: #333; margin: 10px 0; } /* Anpassungen der Tabellenzelle des Paneltitels */ table.recomShell td form table td.title { font-weight: bold; color: #fff; padding: 1px 4px 1px 4px; background: #5C4714; } /* Definitionen fuer die Paneltabelle */ table.recomShell td form table { background: #E3DEC4; } /* Angaben zu den Tabellenzellen der Feldbezeichner der Formularfelder */ table.recomShell td form table td.prompt { font-size: 11px; } /* Angaben zu den Feldbezeichnern der Formularfelder */ table.recomShell td form table td.prompt span { font-size: 11px; color: #333; } /* << */ /* >> Trenner fuer Content Elemente */ /* Trennelement Container */ div.separatorShell { margin-top: 7px; margin-bottom: 15px; background-color: #9A2534; } /* Trennelement Abstandshalter */ div.separatorShell img { } /* << */ /* >> Seitenstruktur */ /* Seitenstruktur Eintragcontainer */ div.sitemapEntry { } /* Definition der Links fuer normal, besucht und beim ueberfahren mit der Maus*/ div.sitemapEntry table td a { text-decoration: none; } /* Anpassungen beim ueberfahren mit der Maus */ div.sitemapEntry table td a:hover { } /* Element definition fuer die momentan aktive Seite */ div.sitemapEntry table td em { font-style: normal; font-weight: bold; color: #000; } /* Einstellungen fuer das Einrueckungselement */ div.sitemapEntry table td img.bullet { height: 4px; width: 4px; margin: 5px 6px 10px 3px; border: 1px solid #000; background: #ccc; } /* Anpassungen fuer den Pfeil fuer die Seite von der aus man auf die Sitemap gegangen ist*/ div.sitemapEntry table td img.arrow { height: 11px; width: 7px; margin: 1px 0 0 5px; background: url(../xist4c/web/krusekopf/01/img/sitemapArrow.gif) left bottom no-repeat; } /* << */ /* >> Standardeinstellungen */ /* Anpassungen fuer die Seiten Stammtabelle */ table.mShell { width: 100%; height: auto; } /* Anpassungen fuer die Zellen der Stammtabelle */ table.mShell td.msCont { } /*Angaben zum Drucken (Diese Angaben bitte nicht veraendern) */ div.printHeader, div.printButtonShell { display: none; } img.printHeaderSpacer { display: none; } /* Schaltet den Skip Link aus (Wird bei Barriere freien Seiten verwendet) */ div.skipNav { display: none; } /* Globale Ebenen Shell */ div.globalLayerShell { width: 989px; position: absolute; left: auto; top: auto; z-index: 31; } /* Breite der Kopfelemente */ table.headerElements { width: 989px; } /* Anpassen der Topnavigation- und Loginzeile */ div.tNavKont, div.tNavKont div.des1 { width: 989px; } div.tNavKont { width: 989px; position: absolute; left: auto; top: 0; width auto; } /* Topnavigation Designcontainer 1 */ div.tNavKont div.des1 { } /* Topnavigation Designcontainer 2 */ div.tNavKont div.des1 div.des2 { } /* Anpassungen der Topnavigationzelle */ div.tNavKont div.des1 div.des2 table.tNavOuter td.navCol { width: 100%; } /* Einstellungen fuer die Schnelllogin- Schnelllogoutzelle */ div.tNavKont div.des1 div.des2 table.tNavOuter td.loginCol { } /* Einstellungen fuer die Schnelllogin- Schnelllogout Platzhalters */ div.tNavKont div.des1 div.des2 table.tNavOuter td.loginCol img { width: 10px; height: 1px; } /* Einstellungen fuer die Suche und Trail Tabelle*/ table.searchAndTrailBg { width: 989px; background: #dde5ee url(../xist4c/web/krusekopf/01/img/quickSearchBg.jpg) top left repeat-y; display: none; } /* Einstellungen fuer die Schnellsuche Tabellenzelle */ td.quickSearchBg { width: 170px; } /* Platzhalterzelle zwischen QuickSearch und Trail */ td.searchAndTrailSpacerColumn { height: 22px; width: 20px; } /* Platzhalter zwischen Quicksearch und Trail */ img.searchAndTrailSpacer { width: 20px; } /* Einstellungen fuer die Trail Tabellenzelle */ td.trailbg { width: 620px; } /* Platzhalter fuer Navigationszelle */ img.navigationColumnSpacer { width: 170px; } /* Definition der Breite des linken Platzhalters */ img.spacerLeft { width: 1px; } /* Definition des Platzhalters fuer den Inhalt */ img.contentSpacer { display: none; } /* Definition der Breite des rechten Platzhalters */ img.spacerRight { width: 1px; } /* Beseitigt ein Problem im Mozilla Browser */ img.block { display: block; } /* Einstellungen fuer die Basistabelle des Inhalts */ table.contentMainTable { width: 989px; height: auto; background: #fff url(../xist4c/web/krusekopf/01/img/background.gif) top left repeat-y; } /* Definitionen der Platzhalterzelle links*/ td.spacerColumnLeft { } /* Anpassungen fuer die Hauptspalte des Inhalts */ td.contentMainColumn { width: 100%; } /* Container fuer den Inhalt */ div.contentContainer { } /* Anpassungen fuer die Inhaltstabelle */ table.contentTable { width: 100%; } /* Definitionen des Inhaltsbereichs */ td.contentColumn { width: 100%; } td.contentColumn div.contSpcShellStd, td.contentColumn div.contSpcShellBL, td.contentColumn div.contSpcContentPager { padding: 0; } /* Definitionen der Platzhalterzelle rechts*/ td.spacerColumnRight { } /* Anpassungen fuer zweispaltigen Inhalt (linke Spalte, abstand zwischen den Spalten, rechte Spalte) */ table.twoColElShell { width: 100%; } table.twoColElShell td.leftSpc img { width: 190px; } table.twoColElShell td.middleSpc img { width: 12px; } table.twoColElShell td.rightSpc img { width: 190px; } table.twoColElShell td.l { } table.twoColElShell td.m { } table.twoColElShell td.r { } /* Grundeinstellungen fuer einige Element festlegen */ th, td, p { font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; font-size: 13px; } /* Anpassungen fuer die Standard Absaetze */ p { margin-top: 0; } /* Allgemeine Einstellung fuer Aufzaehlungspunkte */ ul { list-style-type: square; } /* Einstellungen fuer Ueberschriften */ h1 { font-size: 15px; margin: 0 0 12px 0; color: #9A2534; } h2 { font-size: 14px; margin: 0 0 10px 0; color: #5B4614; } h3 { font-size: 13px; margin: 0 0 8px 0; color: #5B4614; } h4 { font-size: 13px; margin: 0 0 13px 0; } /*Standard Link einstellungen*/ a { color: #9A2534; text-decoration: none; } a:link, a:visited, a:hover { } a:hover { color: #f90; } /* Formulare */ form { margin: 0; } input, textarea, select { font-family: Arial, Verdana, Helvetica, XHelvetica, sans-serif; background: #fff; border: 1px solid #E3DEC4; font-size: 10px; color: #000; width: 100%; -moz-opacity: 0.6; filter: alpha(opacity=60); } input:hover, input:focus, textarea:hover, textarea:focus, select:hover, select:focus { -moz-opacity: 1; filter: alpha(opacity=100); } select { width: auto; } input.button { border: 0; width: auto; } /* Bilder */ img { border: 0; } /* << */ /* >> Seitentitelelement */ /* Aeussere Elemethuelle */ div.titlesShell { margin: 12px 0; padding: 0 15px; } /* Links ausgerichtete Bildspalte */ div.titlesShell table td.imgL { padding-right: 10px; vertical-align: bottom; } /* Links ausgerichtetes Bild */ div.titlesShell table td.imgL div.img { } /* Rechts ausgerichtete Bildspalte */ div.titlesShell table td.imgR { padding-left: 10px; vertical-align: bottom; } /* Rechts ausgerichtetes Bild */ div.titlesShell table td.imgR div.img { } /* Linke Titelspalte */ div.titlesShell table td.titlesL { vertical-align: bottom; } /* Rechte Titelspalte */ div.titlesShell table td.titlesR { vertical-align: bottom; } /* << */ /* >> Umfragen Uebersicht*/ /* Umfrage Tabelle */ table.surveyOvShell { } /* Definition der Frage auf der Uebersichtsseite */ table.surveyOvShell td div span.question { font-size: 13px; font-weight: bold; } /* Einstellungen fuer den Hinweis ueber das Ende der Umfrage */ table.surveyOvShell td div span.date { font-size: 10px; } /* Umfrage Beschreibung */ table.surveyOvShell td div div.desc { } /* << */ /* >> Umfragen Detail */ /* Umfrage Formular */ form.surveyForm { } /* Anpassungen fuer die Tabellenzelle der Detailfrage */ form.surveyForm td.question { font-weight: bold; color: #339; } /* Einstellungen der Tabellenzelle fuer das Ende der Umfrage */ form.surveyForm td.date { font-size: 10px; } /* Versenden Button */ form.surveyForm td input.button { cursor: pointer; /cursor: hand; width: auto; } /* Ergebnistabelle */ table.surveyResult { } /* Frage auf der Ergebnisseite */ table.surveyResult td.question { font-weight: bold; color: #339; } /* Einstellungen fuer den Hinweis ueber das Ende der Umfrage */ table.surveyResult td.date { padding-top: 10px; font-size: 10px; } /* Anpassen des Teilnehmerzusatz */ table.surveyResult td.participants { } /* Anpassungen der Grafik fuer den Ergebnissbalken */ table.surveyResult td div.gfx { border-left: 1px solid #ffeea8; border-top: 1px solid #ffeea8; border-right: 1px solid #6a5503; border-bottom: 1px solid #6a5503; background: #f90 url("img/surveyGraphic.gif") repeat-x; } /* Anpassungen des Hintergrunds fuer den Ergebnissbalken */ table.surveyResult td.gfxBg { background: #efefef; border: 1px inset #ccc; width: auto; } /* << */ /* >> Randbereich links */ /* Definition des linken Randbereichs */ td.leftBorderCol { padding-left: 4px; } /* Anpassungen der Randbereich Breite und den Abstand von oben*/ td.leftBorderCol div.leBoElShell img.topSpc { display: none; } td.leftBorderCol div.leBoElShell img.bottomSpc { height: 1px; width: 236px; } /* Randbereich Inhaltscontainer */ td.leftBorderCol div.lElCont { } /* << */ /* >> Randbereich rechts */ /* Definition des rechten Randbereichs */ td.rightBorderCol { padding-right: 4px; } /* Anpassungen der Randbereich Breite und den Abstand von oben*/ td.rightBorderCol div.riBoElShell img.topSpc { display: none; } td.rightBorderCol div.riBoElShell img.bottomSpc { height: 1px; width: 181px; } /* Randbereich Inhaltscontainer */ td.rightBorderCol div.rElCont { } /* << */ /* >> Teaser rechts */ /* Einstellung des Teaser Inhaltbereichs */ td.rightBorderCol div.rElCont div.title, td.rightBorderCol div.rElCont div.titleBL { } /* Definition des Teasertitel */ td.rightBorderCol div.rElCont div.title h3, td.rightBorderCol div.rElCont div.titleBL h3 { font-size: 14px; margin: 0; color: #fff; display: block; background: url(../xist4c/web/krusekopf/01/img/teaserheadbg.gif) top left repeat-x; padding: 4px 7px 4px 7px; } td.rightBorderCol div.rElCont div.titleBL h3 { background: transparent; padding: 0; border-bottom: 1px solid #dde5ee; color: #000; } /* Definition des Teaseruntertitel */ td.rightBorderCol div.rElCont div.tDesOut1 h4, td.rightBorderCol div.rElCont div.tDesOut1BL h4 { font-size: 11px; margin: 0 0 3px 0; color: #5C4714; } /* Einstellungen fuer die aeussere Teaserhuelle */ td.rightBorderCol div.rElCont div.tDesOut1, td.rightBorderCol div.rElCont div.tDesOut1BL { } td.rightBorderCol div.rElCont div.tDesOut2, td.rightBorderCol div.rElCont div.tDesOut2BL { } td.rightBorderCol div.rElCont div.tOuter, td.rightBorderCol div.rElCont div.tOuterBL { margin-bottom: 1px; } * html td.rightBorderCol div.rElCont div.tOuter, * html td.rightBorderCol div.rElCont div.tOuterBL { width: 100%; } /* Zusaetzliche Design Kontainer fuer den Teaser Inhaltsbereich*/ td.rightBorderCol div.rElCont div.tOuter div.des1, td.rightBorderCol div.rElCont div.tOuterBL div.des1BL { background: url(../xist4c/web/krusekopf/01/img/teaserbg.gif) top left repeat; } td.rightBorderCol div.rElCont div.tOuterBL div.des1BL { background: transparent; } td.rightBorderCol div.rElCont div.tOuter div.des2, td.rightBorderCol div.rElCont div.tOuterBL div.des2BL { padding: 7px; background: url(../xist4c/web/krusekopf/01/img/teaserbgtop.gif) top left repeat-x; } td.rightBorderCol div.rElCont div.tOuterBL div.des2BL { padding: 0; background: transparent; } td.rightBorderCol div.rElCont div.tOuter div.des1 div.des2 div.des3, td.rightBorderCol div.rElCont div.tOuterBL div.des1BL div.des2BL div.des3BL { } /* Einstellungen fuer die Teaserhuelle */ td.rightBorderCol div.rElCont div.tOuter div.des1 div.des2 div.des3 div.content, td.rightBorderCol div.rElCont div.tOuterBL div.des1BL div.des2BL div.des3BL div.contentBL { } * html td.rightBorderCol div.rElCont div.tOuter div.des1 div.des2 div.des3 div.content, * html td.rightBorderCol div.rElCont div.tOuterBL div.des1BL div.des2BL div.des3BL div.contentBL { width: 100%; } /* Anpassungen des Text Inhaltsbereich */ /* Einstellungen der Bild und Content ausrichtung */ td.rightBorderCol div.rElCont div.teaserImgOuter, td.rightBorderCol div.rElCont div.teaserImgOuterL, td.rightBorderCol div.rElCont div.teaserImgOuterR, td.rightBorderCol div.rElCont div.teaserFxImgOuter { padding-top: 2px; margin-bottom: 3px; } /* Fuer Bild zu Textabstand bei links und rechts umfliessend */ td.rightBorderCol div.rElCont div.teaserImgOuterL { padding-right: 6px; } td.rightBorderCol div.rElCont div.teaserImgOuterR { padding-left: 6px; } td.rightBorderCol div.rElCont div.teaserContOuter { } /* Bild zu Textabstand bei rechts und links freibleibend*/ td.rightBorderCol div.rElCont div.teaserContOuterL { padding-left: 6px; } td.rightBorderCol div.rElCont div.teaserContOuterR { padding-right: 6px; } /* Innerer Bild Container */ td.rightBorderCol div.rElCont div.teaserImgInner { margin-left: auto; margin-right: auto; } /* Bild zu Text im zentrierten Modus */ div.paraCeImgOuter { text-align: center; } td.rightBorderCol div.rElCont p { font-size: 11px; margin: 0 0 10px 0; } /* Anpassungen des XHTML Inhaltsbereich */ td.rightBorderCol div.rElCont, td.rightBorderCol div.rElCont p, td.rightBorderCol div.rElCont ul, td.rightBorderCol div.rElCont ol { font-size: 11px; color: #5C4714; } td.rightBorderCol div.rElCont div.tOuterBL, td.rightBorderCol div.rElCont div.tOuterBL p, td.rightBorderCol div.rElCont div.tOuterBL ul, td.rightBorderCol div.rElCont div.tOuterBL ol { color: #5C4714; } td.rightBorderCol div.rElCont div.tOuter a, td.rightBorderCol div.rElCont div.tOuterBL a { color: #000; } td.rightBorderCol div.rElCont div.tOuter a:hover, td.rightBorderCol div.rElCont div.tOuterBL a:hover { color: #5C4714; } /* Anpassungen des XHTML Inhaltsbereich */ td.rightBorderCol div.rElCont p, td.rightBorderCol div.rElCont ul, td.rightBorderCol div.rElCont ol { margin-top: 0; margin-bottom: 10px; } /* Einstellungen der Abstaende des Mehrlinks */ td.rightBorderCol div.rElCont div.teaserSublinkShell { padding: 1px 3px 1px 3px; text-align: left; } /* Anpassen der Schriftgroesse des "mehr..." Links */ td.rightBorderCol div.rElCont div.teaserSublinkShell span.sL a { font-size: 11px; color: #000; } td.rightBorderCol div.rElCont div.teaserSublinkShell span.sL a:hover { color: #f90; } /* Definition fuer die "mehr..." Link Grafik */ td.rightBorderCol div.rElCont div.teaserSublinkShell span.sL img { width: 10px; height: 12px; vertical-align: text-top; border: 0; background: url(../xist4c/web/krusekopf/01/img/littleTeaserArrow.gif) left 4px no-repeat; } /* Anpassungen der Platzhalterzelle der rechten Teaser Spalte*/ td.tRightSpcCol { display: none; } /* Anpassungen des Platzhalters der rechten Teaser Spalte*/ td.tRightSpcCol img { display: none; } /* << */ /* >> Teaser links */ /* Einstellung des Teaser Inhaltbereichs */ td.leftBorderCol div.lElCont div.title, td.leftBorderCol div.lElCont div.titleBL { } /* Definition des Teasertitel */ td.leftBorderCol div.lElCont div.title h3, td.leftBorderCol div.lElCont div.titleBL h3 { font-size: 12px; margin: 12px 0 0 0; color: #000; display: block; padding: 1px 15px 1px 15px; border-bottom: 1px solid #574313; } td.leftBorderCol div.lElCont div.titleBL h3 { background: transparent; } /* Definition des Teaseruntertitel */ td.leftBorderCol div.lElCont div.tDesOut1 h4, td.leftBorderCol div.lElCont div.tDesOut1BL h4 { font-size: 11px; margin: 0 0 3px 0; } /* Einstellungen fuer die aeussere Teaserhuelle */ td.leftBorderCol div.lElCont div.tDesOut1, td.leftBorderCol div.lElCont div.tDesOut1BL { } td.leftBorderCol div.lElCont div.tDesOut2, td.leftBorderCol div.lElCont div.tDesOut2BL { } td.leftBorderCol div.lElCont div.tOuter, td.leftBorderCol div.lElCont div.tOuterBL { margin-bottom: 1px; } /* Zusaetzliche Design Kontainer fuer den Teaser Inhaltsbereich*/ td.leftBorderCol div.lElCont div.tOuter div.des1, td.leftBorderCol div.lElCont div.tOuterBL div.des1BL { background: #F4F3EA; } td.leftBorderCol div.lElCont div.tOuterBL div.des1BL { background: transparent; } td.leftBorderCol div.lElCont div.tOuter div.des2, td.leftBorderCol div.lElCont div.tOuterBL div.des2BL { padding: 7px 15px; } td.leftBorderCol div.lElCont div.tOuterBL div.des2BL { padding: 0; } td.leftBorderCol div.lElCont div.tOuter div.des1 div.des3, td.leftBorderCol div.lElCont div.tOuterBL div.des1BL div.des3BL { } /* Einstellungen fuer die Teaserhuelle */ td.leftBorderCol div.lElCont div.tOuter div.des1 div.des2 div.des3 div.content, td.leftBorderCol div.lElCont div.tOuterBL div.des1BL div.des2BL div.des3BL div.contentBL { margin-bottom: 5px; } * html td.leftBorderCol div.lElCont div.tOuter div.des1 div.des2 div.des3 div.content, * html td.leftBorderCol div.lElCont div.tOuterBL div.des1BL div.des2BL div.des3BL div.contentBL { width: 100%; } /* Anpassungen des Text Inhaltsbereich */ td.leftBorderCol div.lElCont p { font-size: 11px; margin: 0 0 10px 0; } /* Anpassungen des Text Inhaltsbereich */ /* Einstellungen der Bild und Content ausrichtung */ td.leftBorderCol div.lElCont div.teaserImgOuter, td.leftBorderCol div.lElCont div.teaserImgOuterL, td.leftBorderCol div.lElCont div.teaserImgOuterR, td.leftBorderCol div.lElCont div.teaserFxImgOuter { padding-top: 2px; margin-bottom: 3px; } /* Fuer Bild zu Textabstand bei links und rechts umfliessend */ td.leftBorderCol div.lElCont div.teaserImgOuterL { padding-right: 6px; } td.leftBorderCol div.lElCont div.teaserImgOuterR { padding-left: 6px; } td.leftBorderCol div.lElCont div.teaserContOuter { } /* Bild zu Textabstand bei rechts und links freibleibend*/ td.leftBorderCol div.lElCont div.teaserContOuterL { padding-left: 6px; } td.leftBorderCol div.lElCont div.teaserContOuterR { padding-right: 6px; } /* Innerer Bild Container */ td.leftBorderCol div.lElCont div.teaserImgInner { margin-left: auto; margin-right: auto; } /* Bild zu Text im zentrierten Modus */ td.leftBorderCol div.lElCont div.teaserCeImgOuter { text-align: center; } /* Anpassungen des XHTML Inhaltsbereich */ td.leftBorderCol div.lElCont, td.leftBorderCol div.lElCont p, td.leftBorderCol div.lElCont ul, td.leftBorderCol div.lElCont ol { font-size: 11px; color: #574313; } td.leftBorderCol div.lElCont div.tOuterBL, td.leftBorderCol div.lElCont div.tOuterBL p, td.leftBorderCol div.lElCont div.tOuterBL ul, td.leftBorderCol div.lElCont div.tOuterBL ol { color: #574313; } td.leftBorderCol div.lElCont div.tOuter a, td.leftBorderCol div.lElCont div.tOuterBL a { color: #000; } td.leftBorderCol div.lElCont div.tOuter a:hover, td.leftBorderCol div.lElCont div.tOuterBL a:hover { color: #574313; } /* Anpassungen des XHTML Inhaltsbereich */ td.leftBorderCol div.lElCont p, td.leftBorderCol div.lElCont ul, td.leftBorderCol div.lElCont ol { margin-top: 0; margin-bottom: 10px; } /* Einstellungen der Abstaende des Mehrlinks */ td.leftBorderCol div.lElCont div.teaserSublinkShell { padding: 1px 3px 0 3px; text-align: left; } /* Anpassen der Schriftgroesse des "mehr..." Links */ td.leftBorderCol div.lElCont div.teaserSublinkShell span.sL a { font-size: 11px; color: #000; } td.leftBorderCol div.lElCont div.teaserSublinkShell span.sL a:hover { color: #574313; } /* Definition fuer die "mehr..." Link Grafik */ td.leftBorderCol div.lElCont div.teaserSublinkShell span.sL img { width: 10px; height: 12px; vertical-align: text-top; border: 0; background: url(../xist4c/web/krusekopf/01/img/littleTeaserArrow.gif) left 4px no-repeat; } /* << */ /* >> Special Teaser rechts */ /* Einstellungen fuer die aeussere Teaserhuelle */ td.rightBorderCol div.rElCont div.el_wunschliste div.tOuter div.des2 { background-image: url(../../upload/wunschlistebg_2116.jpg); } td.rightBorderCol div.rElCont div.el_gutscheine div.tOuter div.des2 { background-image: url(../../upload/gutschein_3461.jpg); } td.rightBorderCol div.rElCont div.el_verpackung div.tOuter div.des2 { background-image: url(../../upload/verpackungbg_2117.jpg); } /* Einstellungen fuer die Teaserhuelle */ td.rightBorderCol div.rElCont div.el_wunschliste div.tOuter div.des1 div.des2 div.des3 div.content, td.rightBorderCol div.rElCont div.el_gutscheine div.tOuter div.des1 div.des2 div.des3 div.content, td.rightBorderCol div.rElCont div.el_verpackung div.tOuter div.des1 div.des2 div.des3 div.content { padding-top: 70px; } /* Einstellungen der Abstaende des Mehrlinks */ td.rightBorderCol div.rElCont div.el_nolink div.teaserSublinkShell { display: none; } td.rightBorderCol div.rElCont div.el_nolink div.tOuterBL { margin-bottom: 0; } /* << */ /* >> Special Teaser links */ /* Einstellungen der Abstaende des Mehrlinks */ td.leftBorderCol div.lElCont div.el_nolink div.teaserSublinkShell { display: none; } /* Einstellungen der Abstaende des Mehrlinks */ td.leftBorderCol div.lElCont div.el_braun div.tOuterBL div.des1BL { background: #F4F3EA; } /* << */ /* >>Top Navigation*/ /* Einstellungen fuer die Topnavigationtabelle */ table.tNav { margin-top: 5px; } /* Spacereinstellungen fuer die Topnavigation */ table.tNav td.spcLeft { width: 100%; } table.tNav td.spcRight { } /* Formatiert den Bezeichner fuer die Navigation */ table.tNav td.prompt { display: none; color: #574313; font-weight: bold; font-size: 12px; white-space: nowrap; } /* Spalte fuer ein Topnavigations element */ table.tNav td.navItem { } /* Grundeinstellung der Top Navigation fuer die Zustaende "normal", "im Pfad" und "hier"*/ table.tNav td.navItem span.here, table.tNav td.navItem a.normal, table.tNav td.navItem a.inPath { font-size: 11px; font-weight: normal; color: #574313; } table.tNav td.navItem span.here span, table.tNav td.navItem a.normal span, table.tNav td.navItem a.inPath span { } /* Abweichende einstellungen fuer den Zustand "im Pfad" */ table.tNav td.navItem a.inPath { } /* Folgende Angaben definieren die Reaktion beim ueberfahren mit der Maus */ table.tNav td.navItem a.normal:hover, table.tNav td.navItem a.inPath:hover { color: #f90; } /* Angabe fuer den Zustand "hier" */ table.tNav td.navItem span.here { color: #f90; } /* Einstellung der Trennelemente */ table.tNav td.sep img { height: 11px; margin: 0 7px; background-color: #574313; } /* << */ /* >> Navigationspfad */ /* Einstellungen fuer die Navigationspfad Tabelle */ table.trailShell { } /* Angaben zu den Tabellenzellen des Navigationspfad */ td.trailItemCol { } td.trailSepCol { } /* Gemeinsame Einstellungen fuer Trail items */ table.trailShell td a.hNormal, table.trailShell td a.normal { font-size: 10px; text-decoration: none; } table.trailShell td span.hHere, table.trailShell td span.here { font-size: 10px; font-weight: bold; } /* Anpassungen des Trail Homelinks fuer normalen Link, besuchter Link und Reaktion beim Ueberfahren mit der Maus */ table.trailShell td a.hNormal { } table.trailShell td a.hNormal span { } /* Einstellung fuer die momentan aktive Seite */ table.trailShell td span.hHere { } /* Anpassungen der Trail links fuer normalen Link, besuchter Link und Reaktion beim Ueberfahren mit der Maus */ table.trailShell td a.normal { } /* Einstellung fuer die momentan aktive Seite */ table.trailShell td span.here { } /* Anpassungen fuer das Navigationspfad Trennelement*/ table.trailShell td.sep img { height: 9px; width: 15px; margin: 2px 3px 0 3px; background: url(../xist4c/web/krusekopf/01/img/trailSeperator.gif) top left no-repeat; } /* << */ /* >>Panel with Product Table*/ /* Enstellungen fuer den Panelkontainer */ div.panelProductOuter { margin: 0 0 20px 0; } * html div.panelProductOuter { width: 100%; } /* Panel Titelkontainer */ div.panelProductOuter div.panelTitle { border-bottom: 0; } * html div.panelProductOuter div.panelTitle { width: 100%; } /* Panel Titel */ div.panelProductOuter div.panelTitle h3 { margin: 0; } /* Einstellungen fuer den Inhaltskontainer */ div.panelProductOuter div.desOut1 div.des1 { border: 1px solid #AFB0B2; } div.panelProductOuter div.desOut1 div.des2 { } div.panelProductOuter div.desOut1 div.des2 div.cont { padding: 5px 5px 5px 5px; } /* Einstellungen fuer XHTML-Feld */ div.panelProductOuter div.paragraphProductDataRow { padding: 5px 15px 5px 9px; background-color: #ddd; } * html div.panelProductOuter div.paragraphProductDataRow { width: 100%; } div.panelProductOuter form { margin: 0; padding: 0; } /* Einstellungen fuer Preis Tabelle */ div.panelProductOuter table.priceTable { height: 19px; font-size: 8px; background: #72739A; } div.panelProductOuter table.priceTable td.amountCol, div.panelProductOuter table.priceTable td.amountCol input, div.panelProductOuter table.priceTable td.pricePrompt, div.panelProductOuter table.priceTable td.buttonProductLinkShell, div.panelProductOuter table.priceTable td.buttonProductLinkShell input { white-space: nowrap; font-size: 12px; } /* Einstellungen fuer Menge */ div.panelProductOuter table.priceTable td.amountCol { padding: 1px 5px 0 10px; color: #fff; } div.panelProductOuter table.priceTable td.amountCol input { width: 20px; font-size: 8px; margin: 0 0 2px 5px; padding: 1px 2px 1px 2px; border: 1px solid #ccc; } /* Einstellungen fuer Preis */ div.panelProductOuter table.priceTable td.pricePrompt { color: #fff; padding-right: 5px; } div.panelProductOuter table.priceTable td.price { width: 100%; margin: 0; } div.panelProductOuter table.priceTable td.price div { padding: 6px 0 5px 0; text-align: left; background: #fff; } div.panelProductOuter table.priceTable td.price div span { padding-left: 5px; margin-right: 10px; font-size: 10px; line-height: 10px; font-weight: bold; color: #000; } /* Einstellungen fuer Button */ div.panelProductOuter table.priceTable td.buttonProductLinkShell { padding-left: 1px; padding-right: 1px; } div.panelProductOuter table.priceTable td.buttonProductLinkShell div { padding: 0 0 1px 0; } * html div.panelProductOuter table.priceTable td.buttonProductLinkShell div { width: 100%; } div.panelProductOuter table.priceTable td.buttonProductLinkShell input { width: 115px; height: 21px; margin: 0; padding: 0 3px 1px 25px; color: #000; font-size: 11px; line-height: 9px; border: 1px solid #fff; cursor: pointer; cursor: hand; background: #fff url(../xist4c/web/krusekopf/01/img/productRecomButton.gif) left center no-repeat; } div.floatTerm { clear: both; } /* << */ /* >> photogallery items */ /* top bar */ table.povMain div.tbDes4 { border: 1px solid #666; background: #ccc; font-size: 10px; text-align: center; padding: 2px 5px; } /* image */ table.povMain div.iDes3 { border-left: 1px solid #666; border-right: 1px solid #666; background: #efefef; padding-left: 3px; padding-right: 3px; } table.povMain div.iDes4 { text-align: center; vertical-align: middle; padding-top: 5px; } /* image shell */ table.povMain div.imgShell { vertical-align: middle; text-align: center; margin-left: auto; margin-right: auto; } /* text box */ table.povMain div.textBox { overflow: auto; margin-left: auto; margin-right: auto; } table.povMain div.textBox div.tbInner { padding: 0 0 2px 4px; } /* Title */ table.povMain div.title h4 { margin: 0; margin-bottom: 3px; font-size: 13px; } /* description */ table.povMain div.desc p { font-size: 11px; text-align: left; } /* bottom bar */ table.povMain div.bbDes4 { border: 1px solid #666; background: #ccc; font-size: 10px; text-align: right; padding: 2px 5px; } /* << */ /* >> photogallery elements table mode*/ table.povMain { width: 100%; margin-bottom: 12px; } table.povMain caption { text-align: left; font-size: 13px; font-weight: bold; margin-bottom: 10px; } table.povMain table.tableMode { } table.povMain table.tableMode td.iouter { padding: 10px; } /* << */ /* >> photogallery elements float mode*/ table.povMain ul { margin: 0; padding: 0; } table.povMain ul li { display: block; float: left; } table.povMain ul li { padding: 10px; } /* << */ /* >> Photogallery popup body elements*/ #leftPrevBar, #rightPrevBar { position: absolute; top: 0; left: 0; z-index: 2; height: 100%; width: 10px; background: #ccc; } #leftPrevBar[id="leftPrevBar"], #rightPrevBar[id="rightPrevBar"] { position: fixed; width: auto; } #rightPrevBar { left: auto; right: 0; z-index: 3; } #leftDes1, #rightDes1 { height: 100%; border-right: 1px solid #669; } #rightDes1 { border-right: none; border-left: 1px solid #669; } #cImgOuter1 { padding-left: 200px; padding-right: 200px; height: 100%; background: #999; } #cImgOuter1[id="cImgOuter1"] { padding: 0; } #cIOuter2 { padding-left: 30px; padding-right: 30px; } #cIOuter2, #cIOuter3, #cIOuter4 { height: 100%; } /* << */ /* >>Photogallery popup content elements */ div.thumb { padding: 15px; } div.thumb div.thbInner { text-align: center; vertical-align: middle; } div.thumb div.thbInner img { } div.thumb div.thbDes1, div.thumb div.thbDes2, div.thumb div.thbDes3, div.thumb div.thbDes4 { } div.thumb div.thbDes3 { padding: 5px; background: #ccc; } div.thumb div.thbDes4 { } #cImgOuter1 div.image img { width: 100%; } #cImgOuter1 div.image div.imgD3 { text-align: center; padding: 20px; } #cImgOuter1 div.title h1 { padding-left: 20px; padding-right: 20px; font-size: 16px; margin-bottom: 4px; } #cImgOuter1 div.desc p { padding-left: 20px; padding-right: 20px; } /* << */ /* >> Lightbox */ #lightbox { position: absolute; top: 40px; left: 0; width: 100%; z-index: 999; text-align: center; line-height: 0; } #lightbox a img { border: none; } #outerImageContainer { position: relative; background-color: #fff; width: 250px; height: 250px; margin: 0 auto; } #imageContainer { padding: 10px; } #loading { position: absolute; top: 40%; left: 0%; height: 25%; width: 100%; text-align: center; line-height: 0; } #hoverNav { position: absolute; top: 0; left: 0; height: 100%; width: 100%; z-index: 997; } #imageContainer>#hoverNav { left: 0; } #hoverNav a { outline: none; } #prevLink, #nextLink { width: 49%; height: 100%; background: transparent url(../xist4c/web/krusekopf/01/thirdParty/lightbox2_02/img/blank.gif) no-repeat; /* Trick IE into showing hover */ display: block; } #prevLink { left: 0; float: left; } #nextLink { right: 0; float: right; } #prevLink:hover, #prevLink:visited:hover { background: url(../xist4c/web/krusekopf/01/thirdParty/lightbox2_02/img/prevlabel.gif) left 15% no-repeat; } #nextLink:hover, #nextLink:visited:hover { background: url(../xist4c/web/krusekopf/01/thirdParty/lightbox2_02/img/nextlabel.gif) right 15% no-repeat; } #imageDataContainer { font: 10px Verdana, Helvetica, sans-serif; background-color: #fff; margin: 0 auto; line-height: 1.4em; } #imageData { padding:0 10px; } #imageData #imageDetails { width: 70%; float: left; text-align: left; } #imageData #caption { font-weight: bold; } #imageData #numberDisplay { display: block; clear: left; padding-bottom: 1.0em; } #imageData #bottomNavClose { width: 105px; float: right; padding-bottom: 0.7em; } #overlay { position: absolute; top: 0; left: 0; z-index: 998; width: 100%; height: 500px; background-color: #000; filter:alpha(opacity=60); -moz-opacity: 0.6; opacity: 0.6; } .clearfix:after { content: "."; display: block; height: 0; clear: both; visibility: hidden; } * html>body .clearfix { display: inline-block; width: 100%; } * html .clearfix { /* Hides from IE-mac \*/ height: 1%; /* End hide from IE-mac */ } /* << */ /* >> Mailform */ .mailForm { width: 100%; margin-left: 15px; } .mailForm label { display: block; clear: both; margin: 5px 0 2px 0; } .mailForm input, .mailForm textarea { width: 100%; border: 1px solid #ccc; background: #F4F3EA; font-size: 12px; } .mailForm input.button { width: auto; background: transparent; } .mailForm input.submit { width: auto; margin-top: 20px; font-weight: bold; } /* << */ /* Sonstiges */ /* Designtabelle */ table.destable td, table.destable p, table.destable ul, table.destable li, table.destable a { font-size: 11px; } table.destable a { color: #5C4714; } table.destable a:hover { color: #992534; } /* Logos */ div.co_logos div.cont { text-align: center; } div.co_logos img { margin: 3px 12px; } /* << */ cssutils-1.0/sheets/images/0000755000175000017500000000000012416502415014451 5ustar hugohugocssutils-1.0/sheets/images/example.gif0000666000175000017500000000000012126054542016566 0ustar hugohugocssutils-1.0/sheets/import.css0000666000175000017500000000016112126054542015233 0ustar hugohugo@import "import/import2.css"; .import { /* ./import.css */ background-image: url(images/example.gif) } cssutils-1.0/sheets/1utf.css0000666000175000017500000000010112126054542014572 0ustar hugohugo@charset "utf-8"; @import "1inherit-utf8.css"; /* € */ cssutils-1.0/sheets/import3.css0000666000175000017500000000046312126054542015323 0ustar hugohugo/* import3 */ .import3 { /* from ./import/../import3.css */ background: url(images/example3.gif); background: url(./images/example3.gif); background: url(import/images2/example2.gif); background: url(./import/images2/example2.gif); background: url(import/images2/../../images/example3.gif); }cssutils-1.0/sheets/1inherit-iso.css0000666000175000017500000000010412126054542016231 0ustar hugohugo@import "2inherit-iso.css"; /* 1 inherited encoding iso-8859-1 */ cssutils-1.0/sheets/multiple-values.css0000666000175000017500000000003512126054542017051 0ustar hugohugofoo { a: 1, 2 3, bar 4 } cssutils-1.0/sheets/tigris.css0000666000175000017500000002073112126054542015227 0ustar hugohugo/* contains rules unsuitable for Netscape 4.x; simpler rules are in ns4_only.css. see */ /* colors, backgrounds, borders, link indication */ body { background: #fff; color: #000; } .app h3, .app h4, .tabs td, .tabs th, .functnbar { background-image: url(../images/nw_min.gif); background-repeat: no-repeat; } #toptabs td, #toptabs th { background-image: url(../images/nw_min_036.gif); } #navcolumn .body div, body.docs #toc li li { background-image: url(../images/strich.gif); background-repeat: no-repeat; background-position: .5em .5em; } #search .body div, .body .heading { background-image: none; } .app h3, .app h4 { color: #fff; } .app h3, #banner td { background-color: #036; color: #fff; } body #banner td a { color: #fff !important; } .app h4 { background-color: #888; } .a td { background: #ddd; } .b td { background: #efefef; } table, th, td { border: none } .mtb { border-top: solid 1px #ddd; } div.colbar { background: #bbb; } #banner { border-top: 1px solid #369; } .toolgroup { background: #eee; } .toolgroup .label { border-bottom: 1px solid #666; border-right: 1px solid #666; background: #ccc; } .toolgroup .body { border-right: 1px solid #aaa; border-bottom: 1px solid #aaa; } #mytools .label, #projecttools .label, #admintools .label { background: #fff; border-top: 1px solid #666; border-right: none; border-bottom: none; border-left: 1px solid #666; } #mytools .body, #projecttools .body, #admintools .body { background: #fff; border-top: none; border-right: none; border-bottom: none; border-left: 1px solid #666; } #mytools, #projecttools, #admintools { border-right: 1px solid #aaa; border-bottom: 1px solid #aaa; } #helptext { background: #ffc; } #helptext .label { border-bottom: 1px solid #996; border-right: 1px solid #996; background: #cc9; } #helptext .body { border-bottom: 1px solid #cc9; border-right: 1px solid #cc9; } #breadcrumbs { border-top: 1px solid #fff; background-color: #ccc } #main { border-top: 1px solid #999; } #rightcol div.www, #rightcol div.help { border: 1px solid #ddd; } body.docs div.docs { background: #fff; border-left: 1px solid #ddd; border-top: 1px solid #ddd; } body.docs { background: #eee url(../images/help_logo.gif) top right no-repeat !important; } .docs h3, .docs h4 { border-top: solid 1px #000; } #alerterrormessage { background: url(../images/icon_alert.gif) top left no-repeat !important; } .functnbar { background-color: #aaa; } .functnbar2, .functnbar3 { background: #aaa url(../images/sw_min.gif) no-repeat bottom left; } .functnbar3 { background-color: #ddd; } .functnbar, .functnbar2, .functnbar3 { color: #000; } .functnbar a, .functnbar2 a, .functnbar3 a { color: #000; text-decoration: underline; } #topmodule { background: #ddd; border-top: 1px solid #fff; border-bottom: 1px solid #aaa; } #topmodule #issueid { border-right: 1px solid #aaa; } a:link, #navcolumn a:visited, .app a:visited, .tasknav a:visited { color: blue; } a:link.selfref, a:visited.selfref { color: #555 !important; text-decoration: none; } a:active, a:hover, #leftcol a:active, #leftcol a:hover { color: #f30 !important; } #login a:link, #login a:visited { color: white; text-decoration: underline; } #banner a:active, #banner a:hover { color: #f90 !important; } #leftcol a, #breadcrumbs a { text-decoration: none; } #apphead h2 em { color: #777; } .app th { background-color: #bbb; } .tabs th { border-right: 1px solid #333; background-color: #ddd; color: #fff; } .tabs td { background-color: #999; border-bottom: 1px solid #fff; border-right: 1px solid #fff; } .tabs { border-bottom: 6px #ddd solid; } .tabs th, .tabs th a:link, .tabs th a:visited { color: #555; } .tabs td, .tabs td a:link, .tabs td a:visited { color: #fff; } .tabs a { text-decoration: none; } #toptabs td { border-bottom: 1px solid #666; border-right: 1px solid #333; border-left: 1px solid #036; } #toptabs th { border-left: 1px solid #036; } .axial th { background-color: #ddd; color: black } .alert { color: #c00; } .confirm { color: green; } .info { color: blue; } .selection { background: #ffc; } #login { color: #fff; } h4 a:link, h4 a:visited { text-decoration: underline; color: #fff; } /* font and text properties, exclusive of link indication, alignment, text-indent */ body, th, td, input, select, textarea, h2 small { font-family: Verdana, Helvetica, Arial, sans-serif; } code, pre { font-family: 'Andale Mono', Courier, monospace; } html body, body th, body td, textarea, h2 small, .app h3, .app h4, #rightcol h3, #bodycol pre, #bodycol code { font-size: x-small; voice-family: "\"}\""; voice-family: inherit; font-size: small } html>body, html>body th, html>body td, html>body input, html>body select, html>body textarea, html>body h2 small, html>body .app h3, html>body .app h4, html>body #rightcol h3, html>body #bodycol pre, html>body #bodycol code { font-size: small } small, div#footer td, div#login, div.tabs th, div.tabs td, input, select, .paginate, .functnbar, .functnbar2, .functnbar3, #breadcrumbs td, .courtesylinks, #rightcol div.help, .colbar, .tasknav, body.docs div#toc, #leftcol { font-size: xx-small; voice-family: "\"}\""; voice-family: inherit; font-size: x-small } html>body small, html>body div#footer td, html>body div#login, html>body div.tabs th, html>body div.tabs td, html>body input, html>body select, html>body .paginate, html>body .functnbar, html>body .functnbar2, html>body .functnbar3, html>body #breadcrumbs td, html>body .courtesylinks, html>body #rightcol div.help, html>body .colbar, html>body .tasknav, html>body.docs #toc, html>body #leftcol { font-size: x-small } #bodycol h2 { font-family: Tahoma, Verdana, Helvetica, Arial, sans-serif; font-size: 1.5em; font-weight: normal; } .tabs td, .tabs th, dt, .tasknav .selfref, #login .username, .selection { font-weight: bold } h4 { font-size: 1em; } #apphead h2 em { font-style: normal; } /* box properties (exclusive of borders), positioning, alignments, list types, text-indent */ #bodycol h2 { margin-top: .3em; margin-bottom: .5em; } p, ul, ol, dl { margin-top: .67em; margin-bottom: .67em; } h3, h4 { margin-bottom: 0; } form { margin-top: 0; margin-bottom: 0; } #bodycol { padding-left: 12px; padding-right: 12px; width: 100%; voice-family: "\"}\""; voice-family: inherit; width: auto; } html>body #bodycol { width: auto; } .docs { line-height: 1.4; } .app h3, .app h4 { padding: 5px; margin-right: 2px; margin-left: 2px; } .h3 p, .h4 p, .h3 dt, .h4 dt { margin-right: 7px; margin-left: 7px; } .tasknav { margin-bottom: 1.33em } div.colbar { padding: 3px; margin: 2px 2px 0; } .tabs { margin-top: .67em; margin-right: 2px; margin-left: 2px; } .tabs td, .tabs th { padding: 3px 9px; } #toptabs { margin: 0; padding-top: .67em; padding-left: 8px; } #breadcrumbs td { padding: 2px 8px; } #rightcol div.www, #rightcol div.help { padding: 0 .5em } body.docs #toc { position: absolute; top: 15px; left: 0px; width: 120px; padding: 0 20px 0 0 } body.docs #toc ul, #toc ol { margin-left: 0; padding-left: 0; } body.docs #toc li { margin-top: 7px; padding-left: 10px; list-style-type: none; } body.docs div.docs { margin: 61px 0 0 150px; padding: 1em 2em 1em 1em !important; } .docs p+p { text-indent: 5%; margin-top: -.67em } .docs h3, .docs h4 { margin-bottom: .1em; padding-top: .3em; } #alerterrormessage { padding-left: 100px; } .functnbar, .functnbar2, .functnbar3 { padding: 5px; margin: .67em 2px; } #topmodule td { vertical-align: middle; padding: 2px 8px } body { padding: 1em; } body.composite, body.docs { margin: 0; padding: 0; } th, td { text-align: left; vertical-align: top } .right { text-align: right !important; } .center { text-align: center !important; } .axial th { text-align: right; } .app .axial td th { text-align: left; } body td .stb { margin-top: 1em; text-indent: 0; } body td .mtb { margin-top: 2em; text-indent: 0; } .courtesylinks { margin-top: 1em; padding-top: 1em } dd { margin-bottom: .67em; } .toolgroup { margin-bottom: 6px } .toolgroup .body { padding: 4px 4px 4px 0; } .toolgroup .label { padding: 4px; } .toolgroup .body div { padding-bottom: .3em; padding-left: 1em; } #banner td { vertical-align: bottom; } #mytools .body, #projecttools .body, #admintools .body { padding-top: 0; } #mytools, #projecttools, #admintools { margin: -4px 0 6px -4px; padding: 6px; border-right: 1px solid #aaa; border-bottom: 1px solid #aaa; } cssutils-1.0/sheets/html20.css0000666000175000017500000000273712126054542015042 0ustar hugohugo BODY { margin: 1em; font-family: serif; line-height: 1.1; background: white; color: black } H1, H2, H3, H4, H5, H6, P, UL, OL, DIR, MENU, DIV, DT, DD, ADDRESS, BLOCKQUOTE, PRE, BR, HR { display: block } B, STRONG, I, EM, CITE, VAR, TT, CODE, KBD, SAMP, IMG, SPAN { display: inline } LI { display: list-item } H1, H2, H3, H4 { margin-top: 1em; margin-bottom: 1em } H5, H6 { margin-top: 1em } H1 { text-align: center } H1, H2, H4, H6 { font-weight: bold } H3, H5 { font-style: italic } H1 { font-size: xx-large } H2 { font-size: x-large } H3 { font-size: large } B, STRONG { font-weight: bolder } /* relative to the parent */ I, CITE, EM, VAR, ADDRESS, BLOCKQUOTE { font-style: italic } PRE, TT, CODE, KBD, SAMP { font-family: monospace } PRE { white-space: pre } ADDRESS { margin-left: 3em } BLOCKQUOTE { margin-left: 3em; margin-right: 3em } UL, DIR { list-style: disc } OL { list-style: decimal } MENU { margin: 0 } /* tight formatting */ LI { margin-left: 3em } DT { margin-bottom: 0 } DD { margin-top: 0; margin-left: 3em } HR { border-top: solid } /* 'border-bottom' could also have been used */ A:link { color: blue } /* unvisited link */ A:visited { color: red } /* visited links */ A:active { color: lime } /* active links */ /* setting the anchor border around IMG elements requires contextual selectors */ A:link IMG { border: 2px solid blue } A:visited IMG { border: 2px solid red } A:active IMG { border: 2px solid lime } cssutils-1.0/sheets/csscombine-1.css0000666000175000017500000000014112126054542016202 0ustar hugohugo@charset "iso-8859-1"; /* combined sheet 1 */ @namespace s1 "uri"; s1|sheet-1 { top: 1px }cssutils-1.0/sheets/page_test.css0000666000175000017500000000005112126054542015672 0ustar hugohugo@page pageStyle { size: 21.0cm 29.7cm; } cssutils-1.0/sheets/all.css0000666000175000017500000000041012126054542014466 0ustar hugohugo/* this sheet should be used in the HTML, deploy.py does resolve * all imports so that only a single CSS is deployed to cut down * the number of HTTP requests */ @import url(1.css); @import url(atrule.css); @import url(simple.css); @namespace a 'x';cssutils-1.0/sheets/2resolve.css0000666000175000017500000000004412126054542015462 0ustar hugohugo@charset "ascii"; @namespace "x"; cssutils-1.0/sheets/bad.css0000666000175000017500000000011112126054542014442 0ustar hugohugoP { color: yellow; background-color: blue; 1rogue: catch me if you can } cssutils-1.0/sheets/v_simple.css0000666000175000017500000000003512126054542015537 0ustar hugohugoh1 { foo: alpha; bar:beta } cssutils-1.0/sheets/test.css0000666000175000017500000001042112126054542014700 0ustar hugohugo /* * This is a test file only - it doesn't represent a usable style sheet, * it is meant only to test the lexer and parser. */ @import url(foo.css); @import "bar.css"; @import url("fineprint.css") print; @import url("bluish.css") projection, tv; @import url(http://www.steadystate.com/primary.css) screen, tv; @import "sounds_good.css" aural; E[class~="hipster"][thing~="bob"]#myid { att1: hi; foo: bar; } E:lang(c)#unique E.hipster#myid { att1: hi; foo: bar; } @media tv, radio { @page { foo: bar; } H1 { fillcolor: blue } } /* @import url(bad_import.css); */ H1, H2 {color:green;background-color:blue} H1, H2 {color: green;background-color: blue} H3, H4 & H5 { color: red; a: red; b: red; c: red; d: red; e: red; f: red; g: red; h: red; i: red; j: red; k: red; l: red; m: red; n: red; o: red; } H6 { color: black } H7 & H8 { color: red; @page { foo: bar } } P[example="public class foo\ {\ private int x;\ \ foo(int x) {\ this.x = x;\ }\ \ }"] { color: red } H1 { color: red; font-style: 12pt } /* Invalid value: 12pt */ P { color: blue; font-vendor: any; /* Invalid prop.: font-vendor */ font-variant: small-caps } EM EM { font-style: normal } H1 { color: red; rotation: 70minutes } IMG { float: left } /* correct CSS2 */ IMG { float: left here } /* "here" is not a value of 'float' */ IMG { background: "red" } /* keywords cannot be quoted in CSS2 */ IMG { border-width: 3 } /* a unit must be specified for length values */ @three-dee { @background-lighting { azimuth: 30deg; elevation: 190deg; } H1 { color: red } } H1 { color: blue !important } P {counter-increment: par-num ! important } H1 {counter-reset: par-num} P:before {content: counter(par-num, upper-roman) ". "} EM { color: #f00 } /* #rgb */ EM { color: #ff0000 } /* #rrggbb */ EM { color: rgb(255,0,0) } /* integer range 0 - 255 */ EM { color: rgb(100%, 0%, 0%) } /* float range 0.0% - 100.0% */ FOO { content: "Argos\t\n" } R1 { a: U+aaaaa-ffffff } R2 { a: U+?????? } R3 { a: U+f????? } R4 { a: U+fffff? } R5 { a: U+ffffff } * {} /* a=0 b=0 c=0 -> specificity = 0 */ LI {} /* a=0 b=0 c=1 -> specificity = 1 */ UL LI {} /* a=0 b=0 c=2 -> specificity = 2 */ UL OL+LI {} /* a=0 b=0 c=3 -> specificity = 3 */ H1 + *[REL=up]{} /* a=0 b=1 c=1 -> specificity = 11 */ UL OL LI.red {} /* a=0 b=1 c=3 -> specificity = 13 */ LI.red.level {} /* a=0 b=2 c=1 -> specificity = 21 */ #x34y {} /* a=1 b=0 c=0 -> specificity = 100 */ DIV > P:first-child { text-indent: 0 } HTML:lang(fr) { quotes: ' ' ' ' } HTML:lang(de) { quotes: '' '' '\2039' '\203A' } :lang(fr) > Q { quotes: ' ' ' ' } :lang(de) > Q { quotes: '' '' '\2039' '\203A' } H1 { bogus: alpha & beta } H2 { foo: inherit; ;; } /* * Counters and stuff */ P {counter-increment: par-num} H1 {counter-reset: par-num} P:before {content: counter(par-num, upper-roman) ". "} H1:before { content: "Chapter " counter(chapter) ". "; counter-increment: chapter; /* Add 1 to chapter */ counter-reset: section; /* Set section to 0 */ } H2:before { content: counter(chapter) "." counter(section) " "; counter-increment: section; } OL { counter-reset: item } LI { display: block } LI:before { content: counters(item, "."); counter-increment: item } H1:before { content: counter(chno, upper-latin) ". " } H2:before { content: counter(section, upper-roman) " - " } BLOCKQUOTE:after { content: " [" counter(bq, hebrew) "]" } DIV.note:before { content: counter(notecntr, disc) " " } P:before { content: counter(p, none) } /* * Content stuff (w/attr function) */ @media aural { BLOCKQUOTE:after { content: url("beautiful-music.wav") } } IMG:before { content: attr(alt) } H1:before { display: block; text-align: center; content: "chapter\A hoofdstuk\A chapitre" } /* * Clip (w/rect function) */ P { clip: rect(5px, 10px, 10px, 5px); } P { clip: rect(5px, -5px, 10px, 5px); } P { font-size: 12pt; line-height: 12pt } P:first-letter { font-size: 200%; font-style: italic; font-weight: bold; float: left; bogus: foo(p1, p2, "hello", url(primary.css)) } SPAN { text-transform: uppercase } P { color: red; font-size: 12pt } P:first-letter { color: green; font-size: 200% } P:first-line { color: blue } cssutils-1.0/sheets/varsimport.css0000666000175000017500000000037612126054542016137 0ustar hugohugo@variables { c1: "c1 imported"; c2: "c2 imported (SHOULD BE OVERWRITTEN)"; } .import1 { content: var(c1); } .import2 { content: var(c2); } .import3 { /* not defined here! */ content: var(c3); } cssutils-1.0/sheets/csscombine-2.css0000666000175000017500000000015712126054542016212 0ustar hugohugo@charset "ascii"; /* combined sheet 2 */ @import "1.css"; @namespace s2 "uri"; s2|sheet-2 { top: 2px; }cssutils-1.0/sheets/t-HACKS.css0000666000175000017500000000124512126054542015017 0ustar hugohugo/* http://centricle.com/ref/css/filters/?highlight_columns=true */ @import 'styles1.css'; @import "styles2.css"; @import url(styles3.css); @import url('styles4.css'); @import url("styles5.css"); @import "null?\"\{"; @import "styles6.css"; a { color: red; voice-family:"\"}\""; voice-family:inherit; color: green; } b { color: red; c\olor:green; } c { color: red; /*/*/color:green;/* */ } /* NS4 only, should not work: */ d { color: green; /*/*//*/color:red;/* */ } e1{content:"\"/*"} e2{color:green} /* THIS SHOULD WORK??? */ /* \*/ div{color:green} /* */ div#test { } head:first-child+body div { } @media all{/* rules */} cssutils-1.0/sheets/cases.css0000666000175000017500000000012312126054542015015 0ustar hugohugo@import url("a"); b{color:red} a{float:left;bogus:foo(a, url(a))} b{color:red} cssutils-1.0/sheets/bundle.css0000666000175000017500000062665712126054542015222 0ustar hugohugo@charset "UTF-8";html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,big,cite,code,del,dfn,em,font,img,ins,kbd,q,s,samp,small,strike,strong,sub,sup,tt,var,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption,tbody,tfoot,thead,tr,th,td{margin:0;padding:0;border:0;font-weight:inherit;font-style:inherit;font-size:100%;font-family:inherit;vertical-align:baseline}body{line-height:1}ol,ul{list-style:none}table{border-collapse:separate;border-spacing:0}caption,th,td{text-align:left;font-weight:normal}blockquote:before,blockquote:after,q:before,q:after{content:""}blockquote,q{quotes:"" ""}.clearfix:after,#content:after{content:".";display:block;height:0;clear:both;visibility:hidden}.clearfix{zoom:1;display:block}body{font:13px/1.231 sans-serif;*font-size:small;*font:x-small}select,input,button,textarea,button{font:99% sans-serif}table{font-size:inherit;font:100%}pre,code,kbd,samp,tt{font-family:monospace;*font-size:108%;line-height:100%}div{position:relative}section{position:relative;display:block}.sane-defaults code{border:1px solid #ddd;background:#f7f7f7;padding:0 2px;font-size:12px}.sane-defaults pre code{border:0;background:0;padding:0;font-size:inherit}.sane-defaults h2,#wrapper .sane-defaults h2{margin:6px 0;padding:10px 0 0;font-weight:normal;font-size:18px;line-height:24px;color:#666}.sane-defaults h3,#wrapper .sane-defaults h3{margin:6px 0;padding:10px 0 0;font-weight:normal;font-size:16px;line-height:21px;color:#666}.sane-defaults h1{margin-bottom:9px;line-height:24px;color:#333}.sane-defaults h4{font-weight:normal;color:#666}.sane-defaults ol{list-style:decimal inside}.sane-defaults p{margin:0 0 9px}.sane-defaults pre{margin:0 0 1em;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;border:1px solid #ddd;background:#f7f7f7;padding:3px 6px;font-size:12px;line-height:16px;white-space:pre-wrap}.sane-defaults ul{list-style:disc inside}.sane-defaults-2{line-height:18px;counter-reset:ordered-list-item}.sane-defaults-2 h1,.sane-defaults-2 h2,.sane-defaults-2 h3,.sane-defaults-2 h4,.sane-defaults-2 h5,.sane-defaults-2 h6{color:#666}.sane-defaults-2 ol,.sane-defaults-2 ul{margin:0;padding:0 0 0 36px;list-style:none}.sane-defaults-2 ol{counter-reset:ordered-list-item}.sane-defaults-2 ol>li:before{position:absolute;margin:0 0 0 -21px;content:counter(ordered-list-item) ".";counter-increment:ordered-list-item}.sane-defaults-2 ul>li:before{position:absolute;margin:-1px 0 0 -12px;font-size:18px;content:"•"}.sane-defaults-2 li{margin:9px 0}.sane-defaults-2 p{margin:9px 0}.sane-defaults-2 pre{margin:9px 0;padding:9px 12px}.sane-defaults-2 img{display:block;max-width:100%}.ie8 .sane-defaults-2 img{max-width:none}@media print{*{color:#000!important}body{background:#fff!important}#tabs,#footer,#search,#repo-menu{display:none}}body{margin:0;background:#fff;padding:0;line-height:16px;font-family:Helvetica,Arial,sans-serif;color:#393939}p{line-height:18px}a{text-decoration:none;color:#2b547d}a:hover{text-decoration:underline}table{border-spacing:3px}h3,strong{font-weight:bold}em{font-style:italic}.leading{line-height:1.4em}.hidden{display:none}table input,select{margin-bottom:1em}.column ul{margin-top:10px;margin-left:15px;list-style:disc outside none}#introduction_{background:#035;background:-webkit-gradient(linear,left top,left bottom,from(#036),to(#035));background:-webkit-linear-gradient(#036,#035);background:-moz-linear-gradient(top,#036,#035);background:-o-linear-gradient(#036,#035);overflow:hidden}#header,#content,#footer,#introduction_>div{margin:0 auto;width:930px}#introduction_>div>div{float:left}#introduction_ li{line-height:19px}#header-wrap{background:#fff url() repeat-x 0 -6px}#header{position:relative;height:70px}#logo{display:block;float:left;margin-top:17px;width:127px;height:40px;background:url(../images/logo.png) 0 0;line-height:99em;overflow:hidden}#global-nav{position:absolute;right:0;top:5px}#global-nav li{float:left;margin-left:8px}#global-nav li a{background-image:url(../images/global-header-icons.png);background-repeat:no-repeat;padding-left:18px;font-size:85%;color:#8a8a8a}#global-nav li a:hover{text-decoration:none;color:#2b547d}#global-nav a.home{background-position:0 0}#global-nav a.home:hover{background-position:0 -28px}#global-nav a.docs{background-position:0 -57px}#global-nav a.docs:hover{background-position:0 -85px}#global-nav a.support{background-position:0 -117px}#global-nav a.support:hover{background-position:0 -145px}#global-nav a.blog{background-position:0 -177px}#global-nav a.blog:hover{background-position:0 -205px}#global-nav a.plugins{background-position:0 -236px}#global-nav a.plugins:hover{background-position:0 -264px}#global-nav a.forums{background-position:0 -296px}#global-nav a.forums:hover{background-position:0 -324px}#main-nav{float:right;margin-top:43px}#main-nav li{float:left;margin-right:4px;position:relative}#main-nav li a{padding:10px;font-size:108%}#main-nav>ul>li>a{color:#393939}#main-nav li.active,#main-nav #repositories-dropdown.active .drop-arrow{font-weight:bold}#main-nav #repositories-dropdown.active{font-weight:inherit}#main-nav li a span{font-size:85%;color:#6f6f6f}#main-nav>ul>li>a:hover,#main-nav>ul>li>a:hover>span{text-decoration:none;color:#036}#repositories-dropdown>div{position:absolute;width:256px;top:26px;left:-11px;padding:0 4px 4px 4px;overflow:hidden}#repositories-dropdown>div>div{-webkit-border-bottom-left-radius:5px;-webkit-border-bottom-right-radius:5px;-moz-border-bottom-left-radius:5px;-moz-border-bottom-right-radius:5px;border-bottom-left-radius:5px;border-bottom-right-radius:5px;border:1px solid #ccc;border-top:0;background:#fff;-webkit-box-shadow:0 1px 3px rgba(0,0,0,0.2);-moz-box-shadow:0 1px 3px rgba(0,0,0,0.2);box-shadow:0 1px 3px rgba(0,0,0,0.2);font-size:13px;line-height:16px;color:#333}#repositories-dropdown>div>div a{position:relative;display:block;padding:0 0 0 44px;font-size:inherit;line-height:28px;white-space:nowrap}#repositories-dropdown>div>div a:hover{background-color:#fff6c2;text-decoration:none}#repositories-dropdown>div>div a:after{position:absolute;right:0;top:0;bottom:0;display:block;width:16px;border-right:8px solid #fff;background:-webkit-gradient(linear,0 0,100% 0,from(rgba(255,255,255,0)),to(rgba(255,255,255,1)));background:-webkit-linear-gradient(left,rgba(255,255,255,0),rgba(255,255,255,1));background:-moz-linear-gradient(left,rgba(255,255,255,0),rgba(255,255,255,1));background:-o-linear-gradient(left,rgba(255,255,255,0),rgba(255,255,255,1));content:""}#repositories-dropdown>div>div a:hover:after{border-right-color:#fff6c2;background:-webkit-gradient(linear,0 0,100% 0,from(rgba(255,246,194,0)),to(rgba(255,246,194,1)));background:-webkit-linear-gradient(left,rgba(255,246,194,0),rgba(255,246,194,1));background:-moz-linear-gradient(left,rgba(255,246,194,0),rgba(255,246,194,1));background:-o-linear-gradient(left,rgba(255,246,194,0),rgba(255,246,194,1))}#repositories-dropdown>div>div a>img{position:absolute;left:16px;top:4px;display:block;width:16px;height:16px;border:1px solid #ddd;background:#fff;padding:1px}#main-nav ul{float:right}#repositories-dropdown{margin-right:-4px}#repositories-dropdown a.drop-arrow{background:url(../images/drop-arrows.png) no-repeat 99px 14px;padding-right:25px}#repositories-dropdown.active a.drop-arrow{background-position:107px 14px}#repositories-dropdown.inertial-hover>a{margin-left:-6px;margin-right:-6px;padding-left:16px;padding-right:31px}#repositories-dropdown.inertial-hover.hovering>a{margin:-1px -7px;-webkit-border-top-left-radius:5px;-webkit-border-top-right-radius:5px;-moz-border-top-left-radius:5px;-moz-border-top-right-radius:5px;border-top-left-radius:5px;border-top-right-radius:5px;border:1px solid #ccc;background-color:#fff}#repo-overview h4{margin:0;padding:3px 0 6px;font-size:inherit;color:inherit}#repo-overview ol{margin:0 -16px}#repo-overview ol li{float:none;margin:0;height:28px;overflow:hidden}#repo-overview ol li a span{font-size:inherit;color:#999}#main-nav .inertial-hover>div{display:none}#main-nav .inertial-hover>div:hover,#main-nav .inertial-hover.hovering>div{display:block;z-index:3}#repositories-dropdown .group{border-top:1px solid #ddd;padding:12px 16px}#repositories-dropdown .new-repository{margin:0 -16px;background:url(../img/icons/fugue/plus.png) no-repeat 18px;padding:0 16px 0 44px;text-transform:lowercase}#repositories-dropdown .import-repository{margin:0 -16px;background:url(../img/icons/fugue/import.png) no-repeat 18px;padding:0 16px 0 44px;text-transform:lowercase}#main-nav li.search-box{margin:0;width:146px;height:20px}#main-nav #searchbox{position:relative;top:-3px;width:117px;border:1px solid #cfcfcf;background:#fff url(../images/search-icon.png) no-repeat 4px;color:#393939}body.non-webkit #main-nav #searchbox{padding:3px 0 3px 23px}#main-nav #searchbox.placeholder{color:#999}h1{font-weight:bold;font-size:20px;line-height:28px;color:#2b547d}#introduction_{position:relative}#introduction_ h1{margin:0;margin-bottom:8px;border:0;padding:28px 0 15px;text-align:left;font-weight:bold;font-size:300%;line-height:30px;color:#c5efff}#introduction_ h2{padding-bottom:10px;font-weight:bold;font-size:200%;margin-bottom:20px;color:#c5efff}#introduction_ ul{margin-top:0;margin-left:0;list-style:none}#introduction_ li{margin-bottom:7px;background:url(../images/yellow-bullet.png) no-repeat 0 50%;padding-left:15px;font-size:123.1%;color:#fff}#introduction_ .split-column{margin-bottom:10px}#introduction_ .button-wrap{margin:0 auto 15px;width:230px}#introduction_ .column{margin-right:0;margin-bottom:50px;width:650px}#introduction_ .last{width:250px}#introduction_ h2,#plans-and-billing h2{border-bottom:0}#introduction_ #signup-button{margin-top:0}#introduction_ #signup-button a{display:block;margin-left:-100px;width:250px;height:46px;background:url(../images/yellow-button-large.png) no-repeat 0 -75px;padding:29px 0 0 48px;font:bold 24px/16px Helvetica,Arial,sans-serif;color:#173562}#introduction_ #signup-button a:hover{background-position:0 0;text-decoration:none;color:inherit}.atl-button{display:block;background:url(../images/small-blue-button-left.png) no-repeat left top}.atl-button:hover{text-decoration:none}.atl-button>span{display:block;background:url(../images/small-blue-button-right.png) no-repeat right top;padding:0 16px 0 0;line-height:22px}.atl-button>span>span{background:url(../images/blue-arrow.png) no-repeat right 60%;padding:0 12px 0 0;color:#fff}.two-column,.three-column{overflow:visible}#content .three-column{margin-bottom:20px}#content{padding:30px 25px 180px}#content h2{font-weight:bold;font-size:146.5%;color:#2b547d}#content #headline{border-bottom:0}#content h3{color:#2b547d}.two-column .column{display:inline;float:left;margin-right:25px;width:450px}.column.last,.three-column .column.last{float:right;margin-right:0}.three-column .column{display:inline;float:left;margin-right:30px;width:286px}.split-column{overflow:hidden}.split-column .first,.split-column .last{float:left;width:50%}.plan-choices{margin-top:18px;overflow:hidden}.plan-choices .plan{position:relative;display:inline;float:left;margin-right:18px;width:140px;min-height:135px;background:url(../images/plan-box-bottom.png) no-repeat center bottom;text-align:center}#content .plan-choices .featured{margin-top:-10px;min-height:167px;background:url(../images/featured-plan-bottom.png) no-repeat center bottom}.plan-choices .featured h3{background:url(../images/featured-plan-top.png) no-repeat center top}.plan-choices span.offer-terms{position:relative;top:-6px;display:block;font-size:93%;font-size:11px}.plan-choices span.date{font-weight:bold;color:#2b547d}#content .plan h3{margin-top:0;background:url(../images/plan-box-top.png) no-repeat center top;padding-top:10px;font-weight:bold;font-size:161.6%;color:#065a9b}#content .plan-choices .last h3{font-size:124.5%}#content .plan p{color:#414141}#content .plan p.free{position:absolute;left:46px;bottom:51px;display:inline;padding-left:10px;font-size:123.1%}#content .plan p.price{position:absolute;left:46px;bottom:51px;display:inline;background:url(../images/dollar-sign.png) no-repeat left top;padding-left:10px;font-size:146.5%}#content .plan p.price span{font-size:12px}.plan a.atl-button{position:absolute;left:26px;bottom:10px;background:url(../images/small-blue-button-left.png) no-repeat top left;padding-left:16px}.plan a.atl-button:hover{background-position:left -37px}#content .featured a.atl-button{bottom:20px}.plan a.atl-button:hover>span{background-position:right -37px}.plan .current-plan{position:absolute;left:10px;bottom:11px;width:118px;background:url(../images/current-plan-bg.png) no-repeat scroll 0 -1px;padding-bottom:2px;font-weight:bold;font-size:85%}.featured .current-plan{bottom:21px}#content .plan h3 span{display:block;font-weight:normal;font-size:93%}.plan-choices .plan.last{margin-right:0}.plans-included{float:right;font-size:116%;color:#065a9b}#footer{clear:both;margin-top:-101px;height:80px;border-top:1px solid #c1c1c1;padding:10px 0;color:#414141}#footer li{display:inline;height:16px;line-height:16px}#footer-nav li+li:before{padding:0 6px;content:"\0020|\0020"}#footer-nav a{color:inherit}#social-nav{margin:-16px 0 12px 730px;width:200px;text-align:right}#social-nav li{margin-left:6px}#social-nav a:hover{text-decoration:none}#social-nav li.blog a{background:url(../images/rss-icon.png) no-repeat right 1px;padding-right:15px}#social-nav li.twitter a{background:url(../images/twitter-bird-small.png) no-repeat right 3px;padding-right:20px}#footer h5,#technologies{display:inline;font-size:11px;color:#888}#technologies li+li:before{padding:0 4px;content:"\0020/\0020"}#technologies a{color:inherit}table th{vertical-align:top}body.plans #content #features div.first{margin-left:0}.site-message{background:#e7f9e0;text-align:center;height:30px}.site-message p{display:inline-block;background:url(../images/green-upgrade-arrow-icon.png) no-repeat 0 50%;padding-left:26px;line-height:30px;color:#2d6b00}.site-message a,.site-message a:hover,.site-message a:visited{color:#2d6b00;text-decoration:underline}.site-message a:hover{text-decoration:none}.center-blocks{margin:0 auto;width:465px}#search-documentation{width:396px;border:1px solid #c3dfea;background:url(../images/search-docs-bg.png) no-repeat left;padding:8px 0 9px 40px;font-size:153.9%}.delete-list ul.errorlist li{border-bottom:0}.login-box{min-height:150px}.login-box .standard div{padding:5px 0}.login-box .standard .error.message{margin-top:-14px;padding:10px 20px 10px 50px}.login-box .standard label{display:inline-block;width:120px;padding-right:6px;line-height:30px}.login-box .standard label:after{content:":"}.login-box form div ul.errorlist,.login-box .standard .primary-button{margin-left:132px}.login-box .standard th{width:120px}#content .all-plans-include ul li{margin-bottom:12px;font-size:18px;color:#1378d5}.all-plans-include ul li span{color:#393939}.signup .standard .extra-info{margin-top:-14px;margin-left:-20px;margin-bottom:18px;width:418px;border-bottom:1px solid #ccc;background:#dff7d5;padding:9px 15px 9px 20px;color:#2d6b00}.signup .standard .extra-info p{line-height:17px}.redirect-to-atl{background:url(../images/redirect-bb-to-atl.jpg) no-repeat 50%;text-align:center}.redirect-to-atl .redirection{padding:125px 0 100px}.redirect-to-atl .redirection p{font-weight:bold;font-size:20px;line-height:1.5em;color:#2b547d}.follow{background:url(../img/icons/fugue/heart__plus.png) no-repeat 0;padding-left:20px}.follow.following{background-image:url(../img/icons/fugue/heart.png)}.follow.following:hover{background-image:url(../img/icons/fugue/heart_break.png)}.follow.loading{background-image:url(../img/loading.gif)}.tabs{margin-bottom:-1px;padding:0 4px;font-family:"Lucida Grande","Lucida Sans Unicode","Lucida Sans",sans-serif;font-size:12px;color:#525252}.tabs>li{display:block;float:left;margin:0 1px;min-width:60px;-webkit-border-radius:5px 5px 0 0;-moz-border-radius:5px 5px 0 0;border-radius:5px 5px 0 0;border:1px solid #ddd;background:#fafafa;background:-webkit-gradient(linear,0 0,0 100%,from(#fff),to(#f5f5f5));background:-webkit-linear-gradient(#fff,#f5f5f5);background:-moz-linear-gradient(top,#fff,#f5f5f5);background:-o-linear-gradient(#fff,#f5f5f5)}.tabs>li+li{margin-left:0}.tabs a{display:block;padding:1px 8px;text-align:center;font-weight:bold;font-size:11px;line-height:21px;text-decoration:none;color:inherit;outline:0}.tabs>.selected,.tabs li:hover{background:#fff}.tabs>.selected>a{margin-bottom:-1px;border-bottom:1px solid #fff}.tabs .secondary{float:right}#tabs{position:relative;margin:0 -2px;height:24px;padding:0 2px;z-index:2}#content.repo-desc-hidden #tabs{margin:0;padding:0}.dropdown{position:relative}.dropdown ul{display:none;width:150px}.dropdown.inertial-hover.hovering>ul,.dropdown:hover ul{display:block;position:absolute;top:24px;left:-1px;z-index:10}.dropdown li:first-child a{background:#fff url(../img/icons/fugue/plus.png) no-repeat 123px}.dropdown li a{display:block;margin-top:-1px;border:1px solid #ddd;background:#fff;padding:0 10px;text-align:left;font-weight:normal;line-height:24px}.dropdown li a:hover{text-decoration:underline}html,body{height:100%}#wrapper{min-height:100%}form{margin-bottom:0}input{line-height:16px}input[type="email"],input[type="password"],input[type="search"],input[type="text"],textarea{background:#fff;color:#525252;padding:3px}input[type="radio"],input[type="checkbox"],input[type="submit"],input[type="reset"],input[type="button"],input[type="file"],select{font-size:1em}select{background:#fff;line-height:16px;color:#525252}h1,h2,h3,h4,h5,h6{margin-top:0;padding-top:0;margin-bottom:3px}hr{padding:0;margin:5px 0;height:1px;border:0;border-bottom:1px solid #ddd}code,pre,textarea{font-family:"Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace}pre,textarea{font-size:11px}a img{border:0}fieldset{margin:0;border:0;padding:0}h1 a{color:#525252}div{margin:0;padding:0}p{margin:0;padding:0}table{padding:0;margin:0}th{vertical-align:top;text-align:left}.layout-box{border:1px solid #ddd;padding:15px 24px}.primary-button{border-top:1px solid #ddd;border-left:1px solid #ddd;border-bottom:1px solid #bbb;border-right:1px solid #bbb;background:#2b547d;padding:3px 10px;color:#fff;cursor:default}.primary-button:hover{text-decoration:none}.delete-button{border-top:1px solid #ddd;border-left:1px solid #ddd;border-bottom:1px solid #bbb;border-right:1px solid #bbb;background:#f44;padding:2px 8px;color:#fff}.secondary-button{border-top:1px solid #ddd;border-left:1px solid #ddd;border-bottom:1px solid #bbb;border-right:1px solid #bbb;padding:2px 8px;background:#eee;color:#000}.secondary-button-darkbg{background:#fff;color:#000}.bump-button{float:right;margin-right:14px}table.maintable{margin:12px 0;width:100%;border:1px solid #ddd;border-collapse:collapse}table.maintable th,table.maintable td{border-style:solid;border-color:#ddd;padding:.75em 10px .75em 20px}table.maintable th+th,table.maintable td+td{padding-left:10px}table.maintable td{font:12px/1.5em "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace}table.maintable th{font-weight:bold;line-height:18px;text-shadow:0 1px 0 #fff}table.maintable tbody tr:nth-child(even){background:#fcfcfc}table.maintable th,table.maintable td{border-bottom:1px solid #ddd}#wiki .highlight,.highlighttable .highlight{overflow-x:auto;padding-left:3px;padding-right:3px;width:882px}#wiki .highlight pre a:target,.highlighttable .highlight pre a:target{position:absolute;left:0;display:block;width:100%;height:15px;background:#ffffbe;z-index:-1}ul.bulleted{list-style:disc outside}#id_image+ul.bulleted{margin-bottom:10px}.cancel{display:inline-block;line-height:24px}.def-only>dt{position:absolute;left:-9999px}.editable-title:hover{padding:2px 20px 2px 0;background:url(../img/icons/fugue/pencil_small.png) no-repeat right}.editable-icon{background:url(../img/icons/fugue/pencil_small.png) no-repeat left;padding:2px 0 2px 16px;font-size:.8em;color:#525252}.is-spam{background:#eee url(../img/diagonal_transparent_bg.png) repeat left top}.spam{background:url(../img/icons/fugue/minus_small_circle.png) no-repeat left 65%;padding:2px 0 2px 14px;font-size:.8em;color:#525252}.ham{background:url(../img/icons/fugue/plus_small_circle.png) no-repeat left 65%;padding:2px 0 2px 14px;font-size:.8em;color:#525252}.delete{background:url(../img/icons/fugue/cross_small_circle.png) no-repeat left 65%;padding:2px 0 2px 14px;font-size:.8em;color:#525252}.delete[data-comment-is-spam="false"]{display:none}div.cb{clear:both}.noborder{border:0}.nomargin-bottom{margin-bottom:0}.nowrap{white-space:nowrap}.file-icon{background:url(../img/icons/fugue/document.png) no-repeat right;padding:2px 20px 2px 0}.global-event-wrapper{padding:6px 7px;border:1px solid #ddd;background:#fff}img.avatar{display:block;width:32px;height:32px;border:1px solid #e7e7e7;background:#fff;padding:2px}.left{float:left;margin-right:10px}.right{float:right;margin-left:10px}.info-box-lightgrey{margin-bottom:10px;border:1px solid #ddd;background:#eee;padding:15px 20px;font-size:1em;line-height:1.4em}.inset{margin:0 25px}.scroll-x{overflow-x:auto}a.img{text-decoration:none}#repo-access-pane .warning.message,#repo-group-access-pane .warning.message,#plans-and-billing .warning.message,#account-settings-email-pane .warning.message{position:relative;margin:-11px -20px 0;border-bottom:1px solid #ccc;background:#fff6c2 url(../img/icons/atl/yellow-warning-icon.png) no-repeat 20px;padding:10px 20px 10px 60px;text-align:left}#repo-access-pane .warning.message,#repo-group-access-pane .warning.message{margin-bottom:10px;padding:12px 45px 12px 60px}#account-settings-email-pane .warning.message{margin-bottom:13px}#repo-access-pane .warning.message p,#repo-group-access-pane .warning.message p{line-height:17px}#repo-access-pane .warning.message a[href="#hide"],#repo-group-access-pane .warning.message a[href="#hide"]{position:absolute;right:20px;top:12px;display:block;width:17px;height:17px;background:url() no-repeat 50%;line-height:99px;overflow:hidden;cursor:default}.warning{margin-bottom:10px;border:1px solid #ddd;background:url(../img/icons/orange/warning.gif) no-repeat 15px 15px;padding:15px 40px}.errors{margin-bottom:10px;border:1px solid #ddd;background:#eee url(../img/icons/fugue/exclamation.png) no-repeat 15px 15px;padding:15px 40px}.errors .error{display:block;padding-top:3px}.errorlist{margin:0;padding:0;list-style-type:none}.errorlist ul{padding:0;margin:0;list-style-type:none}.column ul.errorlist{margin-left:0}.errorlist li{margin:5px 0 0;background:url(../img/icons/red/warning.gif) no-repeat left;padding:2px 0 2px 20px;list-style:none}.info-paragraph{padding-bottom:0}#twocols1,#twocols2{float:left;width:455px}#twocols1{margin-right:20px}#repo-menu{margin:0 -2px -1px;height:22px;padding:8px 20px}#content.repo-desc-hidden #repo-menu{display:none}.repo-menu{border:1px solid #ddd;color:#525252}.repo-menu h3{line-height:24px}.repo-menu .share{background-image:url(../img/icons/fugue/card_address.png)}.repo-menu .revoke{background-image:url(../img/icons/fugue/cross.png)}.repo-menu .revoke-group{background-image:url(../img/icons/silver/close.gif)}.repo-menu .rss{background-image:url(../img/icons/fugue/feed.png)}.repo-menu .atom{background-image:url(../img/icons/fugue/atom.png)}.repo-menu .pull-request{background-image:url(../img/icons/fugue/arrow_180.png)}.repo-menu .compare-link{background-image:url(../img/arrow_switch.png)}.repo-menu .fork{background-image:url(../img/icons/fugue/arrow_045.png)}.repo-menu .patch-queue{background-image:url(../img/icons/fugue/bandaid.png)}.repo-menu .source{background-image:url(../img/icons/fugue/download-arrow.png)}.repo-menu li{position:relative;float:left}.repo-menu li ul{position:absolute;left:11px;top:20px;display:none;width:115px;border:1px solid #ddd;z-index:2}.repo-menu li:hover ul{display:block}.repo-menu li li{display:block;background:#fff url(../img/layout/bg_lists_small.png) repeat-x 0 0;padding:0;font-size:13px;line-height:16px}.repo-menu li li:hover{background:#fff}.repo-menu li li+li{border-top:1px solid #ddd}.repo-menu li li a{display:block;padding:5px 8px}.repo-menu>ul>li>a,.repo-menu .metadata>li{background-repeat:no-repeat;background-position:0 50%;padding-left:20px;line-height:22px;color:inherit}.repo-menu>ul>li>a{display:block;margin-left:15px;text-decoration:none}.repo-menu .metadata>li{margin-right:12px}.repo-menu .branches{background-image:url(../img/icons/fugue/arrow_135.png)}.repo-menu .tags{background-image:url(../img/icons/fugue/tags.png)}.repo-menu .metadata ul{width:148px}.repo-menu .branches ul{width:172px}.repo-menu .branches li{width:172px}.repo-menu .tags li{width:148px}#repo-menu .source:after,.repo-menu .metadata>li:after{content:"\0020»"}.repo-menu .metadata a{width:106px;overflow:hidden}.repo-menu .branches a{width:130px}.repo-menu a.menu-compare{position:absolute;right:0;top:0;width:16px;height:16px;background:url(../img/arrow_switch.png) no-repeat 50%;padding:5px;text-indent:-9999px}.repo-menu .downloads a{width:82px;background:url(../img/icons/fugue/document_zipper.png) no-repeat 3px;padding-left:25px}#repo-menu-links{float:right}.repo-menu .inertial-hover.hovering.branches>ul,.repo-menu .inertial-hover.hovering.tags>ul,.repo-menu .inertial-hover.hovering.get-source>ul{display:block}#repo-menu-links-mini{display:none}#content.repo-desc-hidden #repo-menu-links-mini{display:block;float:right}#content.repo-desc-hidden #repo-menu-links-mini.nosource{width:168px}#repo-menu-links-mini>li>a{float:left;margin-left:8px;width:16px;height:22px;background-repeat:no-repeat;background-position:0 50%;padding:0}#repo-menu-links-mini li:hover ul{left:4px}#repo-menu-links-mini .follow,#repo-menu-links-mini a[href="#share"],#repo-menu-links-mini .revoke,#repo-menu-links-mini .revoke-group{text-indent:-9999px}#content #repo-heading{margin:0 0 10px;padding:0;font-weight:normal;font-size:16px;line-height:1.5;color:#999}#content #repo-heading.private{margin-left:-1px;background:url(../img/icons/fugue/lock.png) no-repeat;padding-left:21px}#content #repo-heading .owner-username{color:inherit}#content #repo-heading .repo-name{color:#2b547d}.repo-avatar{margin-left:20px;padding-top:2px}#content.repo-desc-hidden .repo-avatar{display:none}#repo-desc{position:relative;margin-bottom:10px;min-height:28px;background:#fff;padding:8px 24px;font-size:1.1em;line-height:1.4em}#repo-desc pre{padding:0;margin:3px 0}#repo-desc h3{display:inline}#repo-desc h3 span{font-size:.8em}.repo-desc-description{margin:0 0 .5em;width:750px}#content.repo-desc-hidden .repo-desc-description{display:none}.repo-desc-private{background:#fff}#repo-desc-cloneinfo{margin:8px 0 0;border-top:1px solid #ddd;padding:8px 0 4px}#content.repo-desc-hidden #repo-desc-cloneinfo{display:none}#repo-desc-cloneinfo.https a.https,#repo-desc-cloneinfo.ssh a.ssh{font-weight:bold;text-decoration:none;cursor:default}#repo-desc-cloneinfo.https #clone-url-ssh,#repo-desc-cloneinfo.ssh #clone-url-https{display:none}#clone-url-https:before,#clone-url-ssh:before,#clone-url-ssh-git:before{content:"$\0020"}#repo-desc .repo-avatar{position:absolute;top:8px;right:0}#toggle-repo-content{position:absolute;left:0;bottom:0;display:block;width:928px;height:5px;border-top:1px solid #ddd;background:#eee url(../img/repo-toggle-up.png)}#content.repo-desc-hidden #toggle-repo-content{background-image:url(../img/repo-toggle-down.png)}#toggle-repo-content:hover{background-color:#ddd}#sourcelist .name{width:275px}#sourcelist .size{width:100px}#sourcelist .text{width:375px}#sourcelist.mq .text{width:300px}#sourcelist .mq{width:75px;padding-right:0}#sourcelist td.name{width:254px;background:url(../img/icons/fugue/document_text.png) no-repeat 0;padding-left:21px;font-size:12px;font-family:Monaco,"Courier New",monospace}#sourcelist td.dirname{background-image:url(../img/icons/fugue/folder.png)}#sourcelist td.subreponame{background-image:url(../img/icons/fugue/folder__arrow.png)}#source-path h1{font-weight:normal;font-size:18px;color:#999}#view-at-form{float:right}#view-at-input{margin-top:4px;width:140px;background:url(../images/search-icon.png) no-repeat 4px}body.non-webkit #view-at-input{margin-top:3px;padding:3px 0 3px 23px}.info-table{width:100%;border-bottom:1px solid #ddd;background:#fafafa}.info-table th{vertical-align:middle}#shortlogs-changes img{float:left;margin:0 9px 0 0;width:16px;height:16px;border:1px solid #ddd}#shortlogs-changes .newtable .files-added,#shortlogs-changes .newtable .files-modified,#shortlogs-changes .newtable .files-deleted{width:36px;padding:0;text-align:center;color:#282}#shortlogs-changes .newtable .files-modified{color:#b70}#shortlogs-changes .newtable .files-deleted{color:#c33}#shortlogs-changes .newtable .files-deleted{padding-right:24px}#shortlogs-changes .newtable th.files-added,#shortlogs-changes .newtable th.files-modified,#shortlogs-changes .newtable th.files-deleted{background:url(../img/icons/green/add.gif) no-repeat 10px;text-indent:-9999px}#shortlogs-changes .newtable th.files-modified{background-image:url(../img/icons/orange/edit.gif)}#shortlogs-changes .newtable th.files-deleted{background-image:url(../img/icons/red/remove.gif)}#shortlogs-changes .newtable .text{width:478px}#shortlogs-changes .newtable .text>a{color:inherit}#shortlogs-changes .commented{background:url(../img/icons/fugue/balloon_left.png) no-repeat 50%}#shortlogs-changes input[type="radio"]{float:left;margin:3px 3px 3px -16px}.diff-radio input{margin:0;padding:0}#changeset-changed{clear:both;font-size:14px;line-height:24px}#changeset-changed>li{padding-left:25px;background:url(../img/icons/orange/edit.gif) no-repeat 4px}#changeset-changed>li.added{background-image:url(../img/icons/green/add.gif)}#changeset-changed>li.removed{background-image:url(../img/icons/red/remove.gif)}#changeset-changed>li:nth-child(odd){background-color:#f7f7f7}#changeset-changed .diffstat{float:right}#changeset-changed .diffstat>ul{float:right;width:70px;padding:4px;overflow:auto}#changeset-changed .diffstat>ul>li{display:block;float:left;width:14px;background:inherit;text-align:center;font-weight:bold;line-height:14px}#changeset-changed .diffstat>ul>li.added{color:#282}#changeset-changed .diffstat>ul>li.removed{color:#c33}#changeset-changed .diffstat>span{display:block;float:right;font-size:12px;font-family:"Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace}#changeset-diff{margin-top:10px;font-size:1.1em}#changeset-diff div.diff{margin:0 0 10px;border:1px solid #ddd}#changeset-diff p{padding:15px 25px}#changeset-diff .changeset-top-link{background:url(../img/icons/fugue/arrow_skip_090.png) no-repeat left;padding:2px 0 2px 20px}#changeset-diff table{width:100%;border-collapse:collapse;border:1px solid #ddd;font-size:1em}#changeset-diff table.jsudiff{border:0;font-size:11px!important}#changeset-diff table td{margin:0;padding:0 5px}#changeset-diff table td.linenr{width:25px;background:#eee;border-right:1px solid #ddd;text-align:right}#changeset-diff table td:target+td.code,#changeset-diff table td:target+td+td.code{background:#ffffbe}#changeset-diff table td.hashnr{width:80px;border-right:1px solid #ddd;background:#fafafa}#changeset-diff table td.linenr pre{width:25px;text-align:right;letter-spacing:-1px;color:#747474}#changeset-diff table td code{margin:0;padding:0;font-size:11px;line-height:1.5em}#changeset-diff table td.linenr a{color:inherit}#changeset-diff tr.ellipsis{background:#efefef;text-align:center}#changeset-diff tr.linecontext{margin:3px;background:#dde7ef;text-align:left;font:11px "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace;letter-spacing:-1px;color:#747474}#changeset-diff tr.del{background:#fdd}#changeset-diff tr.add,#changeset-diff tr.ins{background:#dfd}#changeset-diff tr.marker{background:#eee}#changeset-diff tr.conflict-a,#changeset-diff tr.conflict-b{background:#fe9}#changeset-diff ins,#changeset-diff del{display:inline-block;text-decoration:none}#changeset-diff ins{background:#afa}#changeset-diff del{background:#faa}strong.conflicts{font-weight:normal;color:#c00}.changesets-date{margin:0 auto;border-bottom:0;background:#fff url(../img/layout/bg_general.png) repeat-x left top;padding:10px 20px;font-weight:bold;font-size:1.1em}.changeset-description{margin-bottom:10px;width:710px;white-space:pre-wrap}.files-affected{float:right;margin:0 -100px 0 0;width:96px;overflow:hidden}.files-affected dt{position:absolute;left:-9999px}.files-affected dd{float:left;width:32px;text-align:center}.files-affected dd[title="Added"]{color:#282}.files-affected dd[title="Modified"]{color:#b70}.files-affected dd[title="Removed"]{color:#c33}#source-summary>p{white-space:pre-wrap}#source-summary+p{margin:0 0 5px 25px}.highlighttable{width:928px;border-collapse:collapse}.highlighttable .linenos{border-right:1px solid #ddd;background:#eee;padding-left:5px;padding-right:5px;text-align:right}.highlighttable .linenodiv{width:30px}.highlighttable .code{width:938px}.highlighttable .code-diff{width:885px;border:1px solid #ddd;padding:0}.highlighttable pre{line-height:1.4em}.removed,.added,.command{margin:0;padding:0}.removed{background:#fdd}.added{background:#dfd}.command{background:#dde7ef}#source-view{margin-top:10px;border:1px solid #ddd}#source-view .header{border-bottom:1px solid #ddd;background:#fafafa;padding:4px 10px;line-height:24px;overflow:auto}#source-view .metadata>li{float:left;margin-right:10px}#source-view .source-view-links{float:right}#source-view .source-view-links>li{display:inline}#source-view .source-view-links>li+li:before{content:"/\0020"}#source-view .source-view-form{display:inline}#source-view .source-view-form select,#source-view .source-view-form input[type="submit"]{margin:auto}#inline-image{padding:14px}#inline-image img{margin-bottom:5px}#source-summary{margin:10px 0;border:1px solid #ddd;background:#f7f7f7;padding:10px 300px 10px 20px;font:12px/1.5em "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace;overflow:hidden}#source-summary p{margin:.25em 0 .5em;width:600px;line-height:inherit}.relations{float:right;margin:0 -280px 0 0;text-align:right}.relations dt{position:relative;left:auto;clear:left;float:left;width:100px;text-align:right;color:#747474}.relations dt:after{content:":"}.relations dd{float:left;margin:0 -0.5em 0 0;padding:0 .5em}#source-summary .metadata{position:relative;margin:.5em 0 0;min-height:34px;padding:2px 0 2px 45px;font-family:Helvetica,Arial,sans-serif}#source-summary .metadata dt{position:absolute;left:-9999px}#source-summary .metadata .avatar{position:absolute;left:-3px;top:0}#source-summary .metadata dd:last-child{color:#747474}#source-summary span{font-family:"Lucida Grande","Trebuchet MS",Tahoma,Arial,sans-serif}#source-summary table{border-collapse:collapse}#source-summary table td{padding:0 4px 4px}.tags{display:inline;float:right;margin:0 5px 0 0}.tags li{display:inline;padding:2px 4px}.button-default,.button-tip{margin-right:3px;padding:2px;white-space:normal;font:11px "Lucida Grande","Trebuchet MS",Tahoma,Arial,sans-serif;color:#000}.paginator{float:right;margin:0;padding:0}#wrapper .paginator{list-style:none}.paginator li{float:left;margin-left:2px}.paginator li:hover{background:#eee}.paginator li a{display:block;padding:2px 5px;font-size:1em;text-decoration:none;color:#525252}.paginator li a.current{color:#454b6c;font-weight:bold}.paginator-bottom{margin-top:5px}.repos-all-pages-bottom .paginator{margin-top:15px}li.hellip{width:1.4em;text-align:center;font-size:.8em}.ac_results{border:1px solid black;background-color:white;padding:0;overflow:hidden;z-index:99999}.ac_results ul{margin:0;width:100%;padding:0;list-style:none}.ac_results li{display:block;margin:0;padding:2px 5px;font:menu;font-size:.85em;line-height:16px;overflow:hidden;cursor:default}.ac_odd{background-color:#eee}.ac_over{background-color:#0a246a;color:#fff}.onecol-form,.onecol-info,.pane{position:relative;margin-bottom:10px;background:#fff;padding:43px 20px 15px}.onecol-form{background:#dde7ef}.onecol-info{width:920px}.pane{border:1px solid #ccc;padding:0 20px 12px}.onecol-form h2,.onecol-info h2{position:absolute;left:0;top:0;width:906px;border:1px solid #ccc;padding:7px 0 5px 20px;text-align:left;font-weight:normal;font-size:14px}.pane h2{margin:0 -20px;padding:7px 0 5px 20px;font-weight:normal;font-size:14px}.onecol-info h2,.pane h2{background:#fff url(../img/wiki/bg-th.png) no-repeat left top}.onecol-form h2{background:#fff url(../img/wiki/bg-th-blue.png) no-repeat left top}.twocol-form,.twocol-info{margin-bottom:10px;width:413px;border:1px solid #ddd;background:#f5f5f5;padding:0 20px 10px}.twocol-form{background:#dde7ef;padding-bottom:15px}.twocol-form>.standard{position:static}.twocol-form textarea{width:280px}.twocol-form table{width:100%;border-collapse:collapse}.twocol-form th{padding-right:10px;font-weight:normal}.twocol-form-right td{text-align:right}.twocol-info.access ul{line-height:24px}.twocol-info.access ul a:before{content:"»\0020"}.twocol-info.api p+p{margin:8px 0 0}.twocol-info p:last-child{padding-bottom:0}.twocol-form h3,.twocol-info h3{margin:0 -20px 11px;border-width:1px 0 1px 1px;border-style:solid;border-color:#fff;border-bottom:1px solid #ccc;padding:6px 0 5px 19px;font-weight:normal;font-size:14px}.twocol-form h3{margin-bottom:14px;background:-webkit-gradient(linear,0 0,0 100%,from(#dde7ef),to(#cfdbe8));background:-webkit-linear-gradient(#dde7ef,#cfdbe8);background:-moz-linear-gradient(top,#dde7ef,#cfdbe8);background:-o-linear-gradient(#dde7ef,#cfdbe8)}.twocol-info h3{background:-webkit-gradient(linear,0 0,0 100%,from(#eee),to(#fff));background:-webkit-linear-gradient(#eee,#fff);background:-moz-linear-gradient(top,#eee,#fff);background:-o-linear-gradient(#eee,#fff)}.feeds{margin-bottom:-10px;padding-left:10px;overflow:auto}.feeds li{float:left}.feeds img{display:block;margin:2px}#newsfeed .feeds+p{clear:both;margin-top:0;font-weight:inherit;line-height:18px}#repo-style-admin-preview{border:1px solid #ddd}#explanation p{font-size:13px;line-height:inherit}#explanation{float:right;width:300px;font-size:14px;line-height:18px}#explanation h2{margin-bottom:4px;font-size:16px;line-height:20px;color:#2b547d}#relevant-repos h2{float:none}#relevant-repos h2+a+p{clear:both;font-size:13px;line-height:inherit}#relevant-repos li.private{background-image:none}#relevant-repos li li a{color:#747474}#relevant-repos .follow.following{background-image:url(../img/icons/fugue/heart.png)}ol.users .label{position:relative;left:auto;clear:left;float:left}ol.users .label:after{margin-right:.25em;content:":"}ol.users .label+dd{float:left}.repositories{clear:both}.repositories li{padding-left:20px}.repositories .private{background:url(../img/icons/fugue/lock.png) no-repeat 0 0}.repositories .public{background:url(../img/icons/fugue/lock_disable_unlock.png) no-repeat 0 0}.repositories h4{margin:0;font-weight:normal;font-size:13px;line-height:18px}.repositories p{line-height:18px;color:#666}.thumbnails{clear:both;overflow:auto}.thumbnails li{display:block;float:left;border:1px solid #ccc}.thumbnails li+li{margin-left:2px}.thumbnails a{display:block}.thumbnails img{display:block;width:35px;height:35px}#delete-repository{border:1px solid #ddd;background:#fff;padding:0 20px 15px}#delete-repository h2{margin:0 -20px 15px;border:0;background:#fff url(../img/wiki/bg-th.png) no-repeat 0 0;padding:4px 20px;font-weight:normal;font-size:14px;line-height:24px}#delete-repository td{margin:0}#delete-repository .errorlist{position:absolute;margin:0 0 0 190px}#delete-repository label{line-height:27px}#account-settings-email-pane .status{float:right;margin-right:25px;width:inherit}#account-settings-email-pane .status.sent,#account-settings-email-pane .status.confirmed{color:#666}#account-settings-add-email div{display:inline}#account-settings-add-email label{position:absolute;left:-9999px}#add-private-user #username,#account-settings-email-input{width:15em}#account-settings-cname{margin-top:15px}#account-settings-sshkeys .mapping-widget{font-family:monospace}#newsfeed{float:left;width:555px;line-height:16px}#newsfeed h2{margin-bottom:4px;font-size:16px;line-height:20px;color:#2b547d}#dashboard #newsfeed h2{float:left}#newsfeed>ol>li{position:relative;border-bottom:1px dashed #ddd;padding:9px 0 9px 54px;font-size:1em}#newsfeed>ol>li:last-child{border:0}#newsfeed li li{background-repeat:no-repeat;background-position:0 5px;padding:2px 0 2px 24px}#newsfeed .commit{background-image:url(../img/icons/fugue/puzzle.png)}#newsfeed .create{background-image:url(../img/icons/fugue/plus.png)}#newsfeed .delete,#newsfeed .pullrequest_comment_deleted,#newsfeed .cset_comment_deleted{background-image:url(../img/icons/fugue/slash.png);font-size:inherit;color:inherit}#newsfeed .file_uploaded{background-image:url(../img/icons/fugue/clipboard__plus.png)}#newsfeed .fork{background-image:url(../img/icons/fugue/arrow_045.png)}#newsfeed .issue_comment{background-image:url(../img/icons/fugue/ticket_pencil.png)}#newsfeed .issue_update{background-image:url(../img/icons/fugue/ticket.png)}#newsfeed .report_issue{background-image:url(../img/icons/fugue/ticket_plus.png)}#newsfeed .start_follow_issue{background-image:url(../img/icons/fugue/issue_following.png)}#newsfeed .start_follow_pullrequest,#newsfeed .start_follow_repo,#newsfeed .start_follow_user{background-image:url(../img/icons/fugue/heart.png)}#newsfeed .stop_follow_issue{background-image:url(../img/icons/fugue/issue_follow.png)}#newsfeed .stop_follow_pullrequest,#newsfeed .stop_follow_repo,#newsfeed .stop_follow_user{background-image:url(../img/icons/fugue/heart_break.png)}#newsfeed .strip{background-image:url(../img/icons/fugue/slash.png)}#newsfeed .wiki_created{background-image:url(../img/icons/fugue/clipboard__plus.png)}#newsfeed .wiki_updated{background-image:url(../img/icons/fugue/clipboard__pencil.png)}#newsfeed .pullrequest_created{background-image:url(../img/icons/fugue/envelope_plus.png)}#newsfeed .pullrequest_updated{background-image:url(../img/icons/fugue/envelope_pencil.png)}#newsfeed .pullrequest_comment_created,#newsfeed .pullrequest_comment_updated,#newsfeed .cset_comment_updated,#newsfeed .cset_comment_created{background-image:url(../img/icons/fugue/balloon_left.png)}#newsfeed .pullrequest_fulfilled{background-image:url(../img/icons/fugue/envelope_arrow.png)}#newsfeed .pullrequest_rejected{background-image:url(../img/icons/fugue/envelope_minus.png)}#newsfeed .pullrequest_superseded{background-image:url(../img/icons/fugue/envelope_exclamation.png)}#newsfeed code{font-size:12px}#newsfeed h3{clear:both;margin:1em 0 0;border-bottom:1px solid #ddd;padding-bottom:5px;font-size:14px;line-height:18px;color:#747474}#newsfeed h4{margin:0;line-height:21px;color:#747474}#newsfeed h4:after{content:"…"}#newsfeed h4 a{color:inherit}#newsfeed p{margin-top:3px;font-weight:bold;line-height:inherit}#newsfeed p code{font-weight:normal}#newsfeed p~p{margin-top:0;font-weight:inherit}#newsfeed .commit p~p{margin:3px 0 0 -11px;background:url() no-repeat 0 6px;padding-left:11px}#newsfeed .avatar{position:absolute;left:0;top:10px}#newsfeed time{display:block;font-size:11px;line-height:14px;color:#747474}#newsfeed .paginator{float:none;margin:10px 0 0 76px}.newsfeed-header{margin-bottom:0;padding-bottom:0}.newsfeed-segment-grouper{margin:10px 0 0;border-bottom:1px solid #ddd;font-size:.9em}.newsfeed-avatar{margin-top:12px}table.newsfeed,table.repository-list{width:100%;line-height:21px}table.repository-list .lined-separator{border-top:1px solid #ddd}table.newsfeed{margin-top:-10px;width:428px}table.newsfeed td{vertical-align:top}table.newsfeed td b{background:#dde7ef;font-weight:bold;color:#333}table.newsfeed a{text-decoration:none}table.newsfeed td.newsfeed-icon{width:20px;padding:4px;vertical-align:top}table.newsfeed td.newsfeed-avatar{padding:4px;width:38px;vertical-align:top}td.lined-separator{border-top:1px dashed #ddd}#search label{font-weight:bold;line-height:24px}#search label[for="q"]:after{content:":"}#sign-in .newform>.error_.message_{margin-top:16px}#sign-in .error_{width:258px}#sign-in input[type="email"],#sign-in input[type="password"],#sign-in input[type="text"],#sign-in input[type="url"]{width:286px}.opensocial,.show-standard-signin{display:none}#openid_form .badges{overflow:auto}#openid_form li{display:block;float:left;margin-right:4px}#openid_form a{display:block;width:24px;height:24px;border:1px solid #ddd;background:url() no-repeat 0;text-indent:-9999px;overflow:hidden}#openid_form a.selected{border-color:#2b547d}#openid_form a[title="Yahoo!"]{background-position:-24px}#openid_form a[title="AOL"]{background-position:-48px}#openid_form a[title="OpenID"]{background-position:-72px}#openid_form a[title="MyOpenID"]{background-position:-96px}#openid_form a[title="LiveJournal"]{background-position:-120px}#openid_form a[title="Flickr"]{background-position:-144px}#openid_form a[title="WordPress"]{background-position:-168px}#openid_form a[title="VeriSign"]{background-position:-192px}#openid_form a[title="ClaimID"]{background-position:-216px}#openid_form .inputs li{display:none}#openid_form .inputs li.selected{display:block}#openid_submit{display:none}#associate-openid #openid_form>.badges>li{background:#fff}#associate-openid #openid_form>.inputs>li>input{width:286px}.search-results-result{margin-bottom:10px;border:1px solid #ddd;padding:15px 20px;font:12px/16px "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace}.search-results-result h4{margin-bottom:3px;font-weight:bold;font-size:1em}.search-results-result ul{margin:0;padding:0;list-style-type:none}.search-results-result ul li{padding:2px 0 2px 20px}.search-results-files li{background:url(../img/icons/silver/document.gif) no-repeat left}.search-results-messages li{background:url(../img/icons/silver/comment.gif) no-repeat left}.search-results-authors li{background:url(../img/icons/silver/user.gif) no-repeat left}.search-results-rev{border:1px solid #ddd;border-bottom:0;background:#eee;padding:10px 20px;font-weight:bold;font-size:1.1em}.repository-description{margin:0;padding:0 0 0 20px}.cset-merge{background:url(../img/merge.png) no-repeat bottom right}div#captcha{border-bottom:1px solid #ddd;padding:10px 0 0 20px}#captcha input{width:80px;border:1px solid #ddd}#captcha p{font-size:11px;padding-top:4px}#captcha h4{margin-bottom:6px}#captcha img{float:left;margin:2px 5px 0 0}.markItUpEditor{border:1px solid #ddd;font:14px/18px monospace}.markItUp .markItUpHeader{margin:5px 0;overflow:auto}#welcome .two-thirds-primary h2{margin-top:18px;font-size:16px}#welcome p{margin-top:9px;margin-bottom:9px}#welcome ul{margin:9px 0 9px 9px;line-height:18px;list-style:disc inside}#welcome .two-thirds-secondary h2{font-size:20px;line-height:28px}#welcome .two-thirds-secondary a.img{position:relative;left:-1px}#filters{border-collapse:collapse}#filters td{padding:4px 8px 4px 0}#filters td img.filter-remove{vertical-align:text-bottom}#filters td.filter-header{width:80px}#filters input[type="text"]{width:150px;padding-top:0;padding-bottom:0;font-size:13px}#filters select{margin:2px 0;width:160px}.delete-list ul{margin:0 0 10px;padding:0;list-style-type:none}.delete-list ul li{border-bottom:1px solid #ddd;padding:4px 2px}.delete-list ul li:hover{background:#ddd}.delete-list ul li form{float:right;margin:0;padding:0}#embedded-link{margin:0 0 0 5px;width:120px}.readonly{margin:7px 0 0;background:#fee4a1;border:1px solid pink;padding:3px 8px}.plans-and-billing h4{margin-bottom:8px;font-weight:bold;font-size:18px}#use-your-own-domain form div{margin:10px 0}label[for="id_cname"]{font-weight:bold}label[for="id_cname"]:after{content:":"}.mapping-widget{margin:10px 0 20px;border-top:1px solid #ccc;background:#fff;line-height:16px;cursor:default}.mapping-widget li{position:relative;border-width:0 1px 1px;border-style:solid;border-color:#ccc;padding:4px 10px}.mapping-widget li:after{content:".";display:block;height:0;clear:both;visibility:hidden}.mapping-widget li div{position:static}.mapping-widget>li>span{display:inline-block;width:236px;overflow:hidden}.mapping-widget a[href="#delete"]{position:absolute;right:10px;top:4px;display:block;width:16px;background:url(../img/icons/fugue/minus_circle.png) no-repeat 0;text-indent:-9999px;cursor:inherit;overflow:hidden}.mapping-widget a[href="#delete"].loading{background-image:url(../img/loading.gif)}.mapping-widget a[href="#delete"].removed{background-image:none}.mapping-widget .children{margin:0 -11px -1px;border-top:1px solid #ccc;background:#f5f5f5}.mapping-widget .children li{padding-left:40px}.mapping-widget .access{position:absolute;right:36px;top:4px;color:#888}.mapping-widget li li .access{top:0;height:24px}.mapping-widget .access li{display:inline-block;border:0;padding:0 4px;-webkit-transition:color .25s linear}.mapping-widget .access li:hover,.mapping-widget .access li.selected{color:#333}.expandable.mapping-widget{line-height:24px}.expandable.mapping-widget b{display:inline-block;padding-left:22px}.expandable.mapping-widget li{padding-top:0;padding-bottom:0}.expandable.mapping-widget li[data-username] b{background:url() no-repeat 3px}.expandable.mapping-widget li.loading b{background-image:url(../img/loading.gif);background-position:0 50%}.expandable.mapping-widget li.revealed b{background-image:url()}.expandable.mapping-widget a[href="#delete"]{top:0}#plans-and-billing-div{position:static}#plans-and-billing-div>p{margin-top:10px}#plans-and-billing-div>.access>p{margin-top:6px;line-height:18px}#plans-and-billing-div>.access>div[id]>h4{margin:15px 0 0;font-size:13px}#plans-and-billing .expandable.mapping-widget{margin-bottom:0}#plans-and-billing .spinner{display:none;height:32px;background:url(../img/loading.gif) no-repeat 50% 100%}#plans-and-billing .spinner.loading{display:block}#add-private-user{padding-top:20px;height:27px}#add-private-user .errorlist li{display:block;margin:0;padding-top:0;padding-bottom:0;line-height:21px}#add-private-user div{display:inline}#add-private-user label{position:absolute;left:-9999px}#add-private-user-input{width:15em}#username-mappings-div{margin-left:auto;margin-right:auto}#username-mappings-div p.link{margin-top:5px}#username-mappings .children span:before{content:"alias:\0020";color:#666}#add-username-mapping{margin-bottom:10px;overflow:auto}#add-username-mapping div{float:left}#add-username-mapping div+div{margin-left:20px}#add-username-mapping label{display:block}#add-username-mapping input{margin-left:0;margin-right:0}#add-username-mapping .primary-button{display:block;margin-top:18px}.split-box{overflow:hidden}.split-box h4{margin-top:3px}#name-form input#name{width:265px}input#id_cname{width:140px}#name-form #choose-write-admin{margin-right:4px;width:72px}.plans-and-billing .split-box{margin:10px 0}.input-row{clear:both;margin-bottom:10px;overflow:hidden}.input-row input[type="checkbox"]{margin-left:0}.site-message{border-bottom:1px solid #ccc;background:#e7f9e0;text-align:center}#delete-user{text-align:center}form .actions{text-align:center}form .actions .primary-action{display:inline;margin:10px auto;-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px;border:0;background:#147;background:-webkit-gradient(linear,left top,left bottom,from(#147),to(#036));background:-webkit-linear-gradient(#147,#036);background:-moz-linear-gradient(top,#147,#036);background:-o-linear-gradient(#147,#036);padding:5px 10px;font-weight:bold;font-size:14px;text-shadow:1px 1px 0 #333;color:#fff;cursor:pointer}form .actions .primary-action:hover{background:#036;background:-webkit-gradient(linear,left top,left bottom,from(#036),to(#025));background:-webkit-linear-gradient(#036,#025);background:-moz-linear-gradient(top,#036,#025);background:-o-linear-gradient(#036,#025)}form .actions .primary-action:active{background:-webkit-gradient(linear,left top,left bottom,from(#025),to(#036));background:-webkit-linear-gradient(#025,#036);background:-moz-linear-gradient(top,#025,#036);background:-o-linear-gradient(#025,#036)}.message_{border:1px solid #ccc;padding:10px 0 10px 36px}.ui-dialog .message_,.twocol-form .message_,.mapping-widget li.messages .message_{border-width:0 0 1px 0}.ui-dialog .message_{margin:-20px -13px 20px}.message_.error_{background-color:#ffeae8}.message_.success_{background-color:#e7f9e0}.message_.warning_{background-color:#fff6c2}.message_.note_{background-color:#eafaff}.twocol-form .message_{margin:-14px -20px 20px;padding-left:56px;padding-right:20px}.message_ h4{margin:0 0 0 -15px;background-repeat:no-repeat;background-position:0 0;padding:4px 0 4px 40px;font-weight:bold;line-height:19px}.twocol-form .message_ h4{margin-left:-40px}.ui-dialog .message_ h4{margin-left:-20px}.message_.error_ h4{background-image:url(../img/icons/atl/red-warning-icon.png);color:#b50000}.message_.warning_ h4{background-image:url(../img/icons/atl/yellow-warning-icon.png);background-position:0 2px}.message_.success_ h4{background-image:url(../img/icons/atl/green-upgrade-arrow-icon.png);color:#2d6b00}.message_.note_ h4{background-image:url(../img/icons/info.png);background-position:6px 4px}.message_.warning_ ol{list-style:decimal outside;padding:0 0 0 18px}.message_.warning_ li+li{margin-top:3px}.message_.tooltip_{position:absolute;right:0;margin-top:5px;width:350px;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;box-shadow:0 2px 4px rgba(0,0,0,0.3);padding-top:5px;padding-bottom:5px;z-index:2}.message_.tooltip_ h4{font-size:11px;font-weight:normal;color:#333}.tooltip-anchor{position:relative}.tooltip-anchor:after{content:"";position:absolute;left:-7px;bottom:-6px;margin-left:50%;width:15px;height:8px;background:url();z-index:3}.message{padding:10px 0;text-align:center;border-width:0 0 1px;border-style:solid}.success.message{border-color:#ccc;background:#e7f9e0;color:#2d6b00}.info.message{border-color:#ddd;background:#dde7ef;color:#2b547d}.info.message p{display:inline;background:url(../img/icons/info.png) no-repeat 6px;padding:5px 0 5px 36px}.success.message p{display:inline;background:url(../img/icons/atl/green-upgrade-arrow-icon.png) no-repeat 6px;padding:5px 0 5px 36px}.error.message{margin:-11px -20px 14px;border-bottom:1px solid #ccc;background:#ffeae8 url(../img/icons/atl/red-warning-icon.png) no-repeat 14px;padding:10px 20px 10px 50px;text-align:left;color:#b50000}#header-messages .message{text-align:center;margin:0}#header-messages .error.message{background:#ffeae8}#header-messages .warning.message{background:#fff6c2}#header-messages .message p{display:inline;padding:10px 0 10px 36px}#header-messages .error.message p{background:url(../img/icons/atl/red-warning-icon.png) no-repeat 6px}#header-messages .warning.message p{background:url(../img/icons/atl/yellow-warning-icon.png) no-repeat 6px}.host-domain p{margin-top:8px}#fork-pane p{margin:0 0 10px}#fork-settings-advanced label[for="id_inherit"]{margin-left:22px;line-height:18px}#fork-settings-advanced #id_inherit{position:absolute;left:0}#boxError{width:500px;margin:0 222px}#boxError div{width:160px;float:left}#boxError li{font-size:14px;color:#35638e;font-weight:normal;line-height:1.5em}#imageError{margin:0 auto}h4{font-size:14px;font-weight:bold;color:#2b547d;margin:25px 0 7px}h4 a.hover{text-decoration:underline}.button-group ::selection{background:transparent}.button-group li{float:left;font-weight:bold;font-size:11px}.ui-dialog .ui-dialog-buttonpane button{font-weight:bold;font-size:11px}.button-group li+li{margin-left:-1px}.ui-dialog .ui-dialog-buttonpane button,.button-group li>a{display:block;height:22px;border:1px solid #ccc;filter:progid:DXImageTransform.Microsoft.Gradient(StartColorStr=#FFFFFF,EndColorStr=#DDDDDD);background:#eee;background:-webkit-gradient(linear,0 0,0 100%,from(#fff),to(#ddd));background:-webkit-linear-gradient(#fff,#ddd);background:-moz-linear-gradient(top,#fff,#ddd);background:-o-linear-gradient(#fff,#ddd)!important;padding:0 9px;line-height:23px;text-shadow:0 1px 0 #fff}.ui-dialog.bb-dialog .ui-dialog-buttonpane button{color:#2b547d;height:24px}.ui-dialog .ui-dialog-buttonpane button:first-child,.button-group li:first-child>a{-webkit-border-radius:5px 0 0 5px;-moz-border-radius:5px 0 0 5px;border-radius:5px 0 0 5px;padding-left:10px}.ui-dialog .ui-dialog-buttonpane button:last-child,.button-group li:last-child>a{-webkit-border-radius:0 5px 5px 0;-moz-border-radius:0 5px 5px 0;border-radius:0 5px 5px 0;padding-right:10px}.ui-dialog .ui-dialog-buttonpane button:only-child,.button-group li:only-child>a{-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px}.ui-dialog .ui-dialog-buttonpane button:focus,.ui-dialog .ui-dialog-buttonpane button:hover,.button-group li>a:focus,.button-group li>a:hover{text-decoration:none;color:#393939;cursor:default}.ui-dialog.bb-dialog .ui-dialog-buttonpane button:active,.button-group>.selected>a,.button-group>.selected>a.icon,.button-group>li>a:active,.button-group>li>a.icon:active{filter:progid:DXImageTransform.Microsoft.Gradient(StartColorStr=#CCCCCC,EndColorStr=#EEEEEE);background:#ddd;background:-webkit-gradient(linear,0 0,0 100%,from(#ccc),to(#eee));background:-webkit-linear-gradient(#ccc,#eee);background:-moz-linear-gradient(#ccc,#eee);background:-o-linear-gradient(#ccc,#eee)!important;color:#393939}.ui-dialog.bb-dialog .ui-dialog-title{margin-left:6px}.ui-dialog.bb-dialog .newform{margin-left:7px}.ui-dialog.bb-dialog .ui-dialog-buttonpane button{margin:-1px 7px 14px}.repo-desc-hidden #fork-actions{display:none}#fork-actions.button-group{float:right}.ui-dialog.bb-dialog .ui-dialog-buttonpane button,.button-group>li>a.icon{background:-webkit-gradient(linear,0 0,0 100%,from(#fff),to(#ddd));background:-webkit-linear-gradient(#fff,#ddd);background:-moz-linear-gradient(top,#fff,#ddd);background:-o-linear-gradient(#fff,#ddd);position:relative;cursor:pointer;padding-left:28px}.ui-dialog.bb-dialog .ui-dialog-buttonpane button:before,.button-group>li>a.icon:before{position:absolute;display:block;width:16px;height:16px;left:9px;top:3px;content:""}@-moz-document url-prefix(){.ui-dialog.bb-dialog .ui-dialog-buttonpane button:before{left:-22px;top:-3px}}#fork-actions.button-group>li>a.compare-link:before{background:url(../img/arrow_switch.png)}#fork-actions.button-group>li>a.pull-request:before,#pullrequest-actions.button-group>li>a.pull-request:before{left:8px;top:4px;background:url(../img/icons/fugue/arrow_180.png)}.ui-dialog.pr-fulfill-dialog .ui-dialog-buttonpane button:before,#pullrequest-actions.button-group>li>a.fulfill:before{background:url(../img/icons/fugue/tick-sprite.png)}.ui-dialog.pr-reject-dialog .ui-dialog-buttonpane button:before,#pullrequest-actions.button-group>li>a.reject:before{background:url(../img/icons/fugue/cross.png)}#pullrequest-actions.button-group>li>a.loading:before{background:url(../img/loading.gif)}#changeset-diff .reply td{padding:0}label[for="id_content"]{position:absolute;text-indent:-9999px}#id_content,#id_message{margin:0 0 6px;width:400px;height:90px;border:1px solid #e7e7e7;padding:3px 3px 3px 6px;font:13px/18px Helvetica,Arial,sans-serif;color:#393939}#id_message{margin-left:40px}.pull-request label[for="id_message"]{display:block;margin-top:10px}.pull-request #id_message{margin-left:0}form.comment{margin:0 0 0 102px;padding:0 0 0 6px}form.comment table{border-spacing:0}tr.disable-row{background:#ccc;font-style:italic}tr.disable-row td,tr.disable-row td a{color:gray}table.info{width:100%;border-collapse:collapse;font-size:14px;line-height:1.5em}table.info th,table.info td{border:1px solid #ddd;padding:3px 9px}table.info th{background:#eee;background:-webkit-gradient(linear,left top,left bottom,from(#f7f7f7),to(#e7e7e7));background:-webkit-linear-gradient(#f7f7f7,#e7e7e7);background:-moz-linear-gradient(top,#f7f7f7,#e7e7e7);background:-o-linear-gradient(#f7f7f7,#e7e7e7);font-weight:bold;text-shadow:0 1px 0 #fff}table.info tr:nth-child(odd){background:#f7f7f7}a.action{display:inline;margin:10px auto;-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px;border:0;background:#147;background:-webkit-gradient(linear,left top,left bottom,from(#147),to(#036));background:-webkit-linear-gradient(#147,#036);background:-moz-linear-gradient(top,#147,#036);background:-o-linear-gradient(#147,#036);padding:5px 10px;font-weight:bold;font-size:14px;text-shadow:1px 1px 0 #333;color:#fff;cursor:pointer}a.action:hover{background:#036;background:-webkit-gradient(linear,left top,left bottom,from(#036),to(#025));background:-webkit-linear-gradient(#036,#025);background:-moz-linear-gradient(top,#036,#025);background:-o-linear-gradient(#036,#025);text-decoration:none}a.action:active{background:-webkit-gradient(linear,left top,left bottom,from(#025),to(#036));background:-webkit-linear-gradient(#025,#036);background:-moz-linear-gradient(top,#025,#036);background:-o-linear-gradient(#025,#036)}td.code a{display:block;padding:0 3px;color:inherit}td.code a:hover,#changeset-diff td.code a:hover del,#changeset-diff td.code a:hover ins{background:#bdf;text-decoration:none}.comments-link{position:relative;top:3px;display:inline-block;margin:3px 0 0 5px;width:16px;height:16px;background:url(../img/icons/fugue/balloon_left.png);text-indent:-99999px;overflow:hidden}td.comments .comments-link{margin-left:0}.cset-comment-count{float:right;margin:0 5px 0 0}td.fulfilled,td.unfulfilled{background:url(../img/icons/fugue/cross.png) no-repeat 50%;text-indent:-9999px}td.fulfilled{background-image:url(../img/icons/fugue/tick.png)}.progress{position:relative;top:-1px;margin:0 -4px;border:1px solid #ccc}.progress div[aria-valuenow]{background:#fff6c2;text-align:center;font-size:11px;line-height:15px;letter-spacing:2px}.progress span{padding:0 3px}#search-form{float:left;margin-bottom:10px}#search-form input{display:block;float:left;margin:0;padding:4px 5px 4px 30px;width:400px;height:20px;font-size:18px;line-height:20px;border:1px solid #ddd;background:url(../images/search-docs-bg-30px.png) 0 -1px no-repeat}#search-form button{display:block;float:left;margin-left:3px;border:1px solid #ddd;height:30px;font-size:15px;padding:0 10px}#search-form+p{clear:both}.subtitle{clear:both;font-size:11px;line-height:20px}#content .repositories{border-bottom:0}.detailed>li{position:relative;min-height:41px;padding:10px 10px 10px 61px;overflow:auto;-webkit-transition:background .2s linear}.detailed>li:nth-child(even){background:#fcfcfc}.detailed>li:hover{background:#fff6c2}dd.name{float:left;font-size:18px;line-height:22px;color:#999}dd.name a:first-child{color:inherit}dd.name a:hover{text-decoration:none}dd.metadata{display:block;clear:left;float:left;color:#999}dd.metadata a{font-weight:bold;color:inherit}dd.avatar{position:absolute;left:10px;top:10px;width:auto;height:auto;background:#fff}dd.avatar a{display:block}dd.avatar img{display:block;width:35px;height:35px}.detailed p{clear:left;float:left;margin-top:6px;line-height:16px}dd.private.repository,dd.public.repository{position:absolute;right:251px;top:14px;width:16px;height:16px;background:url(../img/icons/fugue/lock.png) no-repeat;text-indent:-9999px}dd.public.repository{top:13px;background-image:url(../img/icons/fugue/lock_disable_unlock.png)}dd.commits,dd.forks,dd.followers{position:absolute;top:10px}dd.commits{right:213px}dd.followers{right:124px}dd.forks{right:47px}.info-button{height:22px;font-size:12px;line-height:22px}.info-button .info-button-action{display:block;-webkit-border-radius:5px 0 0 5px;-moz-border-radius:5px 0 0 5px;border-radius:5px 0 0 5px;border:1px solid #ccc;background:#eee;background:-webkit-gradient(linear,left top,left bottom,from(#fff),to(#ddd));background:-webkit-linear-gradient(#fff,#ddd);background:-moz-linear-gradient(top,#fff,#ddd);background:-o-linear-gradient(#fff,#ddd)}.info-button .info-button-action:active{background:-webkit-gradient(linear,left top,left bottom,from(#aaa),to(#eee),color-stop(0.25,#ccc));background:-webkit-linear-gradient(#aaa,#ccc 25%,#eee);background:-moz-linear-gradient(top,#aaa,#ccc 25%,#eee);background:-o-linear-gradient(#aaa,#ccc 25%,#eee)}.info-button-readonly .info-button-action,.info-button-readonly .info-button-action:hover{background:#fff}.info-button .info-button-action a{display:block;width:22px;background-repeat:no-repeat;background-position:6px;padding:0 6px 0 0;text-indent:-9999px;cursor:default}.info-button .info-button-action a{background-image:url(../img/icons/fugue/puzzle.png)}.followers .info-button-readonly .info-button-action a,.followers-not-following .info-button .info-button-action a{background-image:url(../img/icons/fugue/heart_empty.png)}.followers-not-following .info-button .info-button-action a:hover,.followers-following .info-button .info-button-action a{background-image:url(../img/icons/fugue/heart.png)}.followers-following .info-button .info-button-action a:hover{background-image:url(../img/icons/fugue/heart_break.png)}.forks .info-button .info-button-action a{background-image:url(../img/icons/fugue/arrow_045.png)}.followers .info-button-requesting .info-button-action a,.followers .info-button-requesting .info-button-action a:hover{background-image:url(../img/loading.gif)}.info-button .info-button-label{position:absolute;left:29px;display:block;width:36px;-webkit-border-radius:0 5px 5px 0;-moz-border-radius:0 5px 5px 0;border-radius:0 5px 5px 0;border:1px solid #ccc;background:#fff;text-align:center}.commits .info-button .info-button-label{width:48px}button{background:#eee;background:-webkit-gradient(linear,left top,left bottom,from(#fff),to(#ddd));background:-webkit-linear-gradient(#fff,#ddd);background:-moz-linear-gradient(top,#fff,#ddd);background:-o-linear-gradient(#fff,#ddd)}button:active{background:#ccc;background:-webkit-gradient(linear,left top,left bottom,from(#aaa),to(#eee),color-stop(0.25,#ccc));background:-webkit-linear-gradient(#aaa,#ccc 25%,#eee);background:-moz-linear-gradient(top,#aaa,#ccc 25%,#eee);background:-o-linear-gradient(#aaa,#ccc 25%,#eee)}dl.menu{line-height:20px}dl.menu dd{margin-left:10px}dl.menu dd+dt{margin-top:5px}ol.code{margin:8px 0 0;border:1px solid #ddd;background:#eee;padding:4px 12px;font-size:12px;line-height:1.5em}ol.code li{line-height:inherit}.container_{margin:0 25px}p.link a:after{content:"\0020»"}#grant-access.loading,#grant-group-access.loading{background:url(../img/loading.gif) no-repeat 386px}#grant-access div,#grant-access input,#grant-group-access div,#grant-group-access input{display:inline}#grant-access label,#grant-group-access label{position:absolute;left:-9999px}#grant-access-user-input{width:15em}label[for="access-level"],label[for="group-access-level"]{position:absolute;left:-9999px}#access-level,#group-access-level{margin:2px}#repo-group-access-pane>p{margin-bottom:5px}#group-access{margin-bottom:10px}#grant-group-access.loading{background:url(../img/loading.gif) no-repeat 386px 6px}#grant-group-access #groupname{width:200px}#what-is-bitbucket{display:none}.new-to-bitbucket{position:relative;margin-bottom:30px;border:1px solid #adf;background:#f2faff;background:-webkit-gradient(linear,left top,left bottom,from(#fafaff),to(#eafaff));background:-webkit-linear-gradient(#fafaff,#eafaff);background:-moz-linear-gradient(top,#fafaff,#eafaff);background:-o-linear-gradient(#fafaff,#eafaff);padding:20px 20px 20px 75px;color:#414141}.new-to-bitbucket .avatar{position:absolute;left:20px;top:20px}#content .new-to-bitbucket h2{margin:0;padding:0;font-size:16px;line-height:20px;color:#065a9b}.new-to-bitbucket p{margin-top:2px;line-height:16px}.primary-action-link{display:inline-block;*display:inline;zoom:1;height:36px;-webkit-border-radius:8px;-moz-border-radius:8px;border-radius:8px;background:#147;padding:1px}.new-to-bitbucket .primary-action-link{position:absolute;right:20px;top:20px}.new-to-bitbucket .actions .primary-action-link{position:static;right:0;top:0;margin:20px 20px 0}.primary-action-link a{display:block;height:34px;-webkit-border-radius:7px;-moz-border-radius:7px;border-radius:7px;border-width:1px;border-style:solid;border-color:#369;border-top-color:#47a;background:#258;background:-webkit-gradient(linear,left top,left bottom,from(#369),to(#147));background:-webkit-linear-gradient(#369,#147);background:-moz-linear-gradient(top,#369,#147);background:-o-linear-gradient(#369,#147);padding:0 12px 0 18px;font-weight:bold;font-size:16px;line-height:32px;color:#fff;text-shadow:0 -1px 0 #333}.primary-action-link a:after{position:relative;top:1px;font-size:20px;font-family:Arial,sans-serif;content:"\0020»";color:rgba(255,255,255,0.8)}.primary-action-link a:hover{border-top-color:#369;background:#147;background:-webkit-gradient(linear,left top,left bottom,from(#258),to(#036));background:-webkit-linear-gradient(#258,#036);background:-moz-linear-gradient(top,#258,#036);background:-o-linear-gradient(#258,#036);text-decoration:none}.primary-action-link a:active{border-color:#258;background:#036;background:-webkit-gradient(linear,left top,left bottom,from(#036),to(#258));background:-webkit-linear-gradient(#036,#258);background:-moz-linear-gradient(top,#036,#258);background:-o-linear-gradient(#036,#258);text-decoration:none}.new-to-bitbucket a[href="#hide"]{position:absolute;left:3px;top:3px;display:none;width:17px;height:17px;background:url() no-repeat 50%;text-indent:-9999px;cursor:default}.jira{background:url(../img/icons/fugue/sticky_notes__arrow.png) no-repeat 0;padding-left:20px}#invitation-login{display:none}#invitation-dialog{padding-top:20px}#invitation-dialog .error_.message_,#invitation-dialog .success_.message_{display:none}#invitation-dialog input{width:265px}#invitation-dialog select{margin-left:15px}#invitation-blurb a[href="#share"]{background:url(../img/icons/fugue/card_address.png) no-repeat 0;padding-left:20px}#repo-access-pane #invitation-blurb{margin-top:10px}#toggle-email-notifications{float:right;margin-top:3px;margin-right:6px}#toggle-email-notifications .loading{background:url(../img/loading.gif) no-repeat 0 0;padding-left:20px}input.inline-field{border:1px solid #ddd;margin:5px -5px;height:16px;width:100%}input.inline-field.loading{background:url(../img/loading.gif) no-repeat 99% 50%}input.inline-field.placeholder,input.placeholder{color:#999}.mapping-widget li.messages .message_{margin:0 -10px;padding:5px}.mapping-widget li.messages .message_ h4{margin-left:5px;padding-left:30px;font-size:13px;line-height:17px}.unmapped-user{border-bottom:1px dotted #aaa}.unmapped-user a{font-size:.95em}.jsudiff{white-space:pre;font:11px/16px "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace;color:#747474}.jsudiff .context{background:#eee}.jsudiff td{display:block;float:left;height:16px}.jsudiff .number{width:25px;border-right:1px solid #ddd;background:#eee;padding:0 5px;text-align:right}.jsudiff a{color:inherit}.jsudiff .code{color:#393939}.jsudiff .context .code{color:inherit}a.compare-link img{position:relative;top:3px}#groups{position:relative;margin:0 auto;width:413px}#groups .link{margin-top:5px}#groups-container>ol>li{margin:10px -10px 0;padding:0 10px 10px}#groups h2{float:left;margin:0;padding:10px 0 0;font-weight:normal;font-size:18px;line-height:22px;color:#999}#groups h2 span{color:#2b547d}.group-members>.mapping-widget{clear:both;margin:0 0 10px}.group-members>ol{margin-bottom:0}.group-members>.errorlist{clear:both}.group-members form div{float:left}.group-members form div+div{float:right}.group-members label{position:absolute;left:-9999px}.group-members input[type="text"]{width:320px}.group-members input[type="submit"]{margin-right:0}#add-group-link{position:absolute;right:20px;top:0;background:url(../img/icons/fugue/plus.png) no-repeat 0;padding-left:20px;line-height:28px;color:#525252}#add-group-form{display:none;padding-top:20px}#groups .editable-title form{display:inline-block;margin:-4px -1px}#groups .editable-title input{width:220px;padding:0}#groups .delete-group{display:block;clear:both;margin-bottom:10px;color:#999;text-decoration:underline}#groups h2+div{float:right;padding-top:11px}#groups h2+div label{padding-right:2px}#groups h2+div select{margin-bottom:0}#groups-container .group-members{margin:10px -10px 0;padding:0 10px;overflow:auto}#add-group-input{width:265px}td.cset-delimiter{color:gray;text-align:center;font-family:monospace}#loadmoar{display:block;margin-top:16px;border:1px solid #ddd;padding:4px;text-align:center;color:#747474}#loadmoar:hover{background:#f7f7f7;text-decoration:none}#loadmoar.spinner{background:url(../img/loading.gif) no-repeat 50%;text-indent:-9999px}#infobox{position:fixed;right:4px;top:4px;height:10px;width:10px;text-indent:-9999px;border:2px solid #000;-moz-border-radius:8px;border-radius:8px}#infobox:hover{text-indent:0;width:300px;height:65px;border:1px solid #ece4ea;background:rgba(255,255,255,0.8);padding:4px;text-align:right}#infobox h4{margin:0 2px 0 0;border-bottom:1px solid #ece4ea}#infobox dl{margin:8px 0 0 0}.if-orange{background:#ffbf00}.if-red{background:red}.if-green{background:#dfd}#infobox dt,#infobox dd{float:right;margin-right:8px;height:18px}#infobox input{height:8px}.newtable{border-collapse:collapse;line-height:28px;color:#333}.newtable tr{display:block;clear:both}.newtable th,.newtable td{display:block;float:left;border-bottom:1px solid #ddd;height:28px;padding:0 20px 0 0;overflow:hidden}.newtable th{border-bottom-width:2px;font-weight:bold}.newtable th.date,.newtable td.date{width:100px}.newtable td.hash{font-size:12px}.newtable td.hash code{border:0;background:0;padding:0}.newtable th.person,.newtable td.person{width:160px}.newtable td.person{position:relative;width:132px;padding-left:28px}.newtable td.person img,.newtable td.repo img{position:absolute;left:0;top:4px;display:block;width:16px;height:16px;border:1px solid #ddd;padding:1px}.newtable td.repo{position:relative;padding-left:28px;color:#999}.newtable td.repo.deleted{padding-left:0}.newtable td.repo a,.newtable td.repo a:hover{color:inherit}.newtable td.repo a~a,.newtable td.repo a~a:hover{color:#2b547d}.newtable th.labels{position:absolute;left:-9999px}.newtable td.labels{position:absolute;right:140px;background:#fff;padding-right:0;overflow:visible}.newtable td.labels:before{position:absolute;left:-32px;top:0;bottom:0;display:block;width:32px;background:#fff;background:-webkit-gradient(linear,0 0,100% 0,from(rgba(255,255,255,0)),to(rgba(255,255,255,1)),color-stop(0.75,rgba(255,255,255,1)));background:-webkit-linear-gradient(left,rgba(255,255,255,0),rgba(255,255,255,1));background:-moz-linear-gradient(left,rgba(255,255,255,0),rgba(255,255,255,1));background:-o-linear-gradient(left,rgba(255,255,255,0),rgba(255,255,255,1));content:""}.newtable .labels dt{position:absolute;left:-9999px}.newtable .labels dd{display:block;float:left;margin:4px 0 0 4px;max-width:150px;height:18px;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;border:1px solid #ddd;background:#eee;padding:0 6px;text-align:right;white-space:nowrap;font:11px/18px Monaco,monospace;color:#555;overflow:hidden;text-overflow:ellipsis;text-overflow:ellipsis-word}.newtable .labels .tag{background:url(../img/icons/fugue/tags.png) no-repeat 4px;padding-left:24px}.newtable .labels.primary-branch:before,.newtable .labels.primary-branch .branch{display:none}.two-thirds-primary{float:left;width:555px}.two-thirds-secondary{float:right;width:300px}.two-thirds-secondary>h3{padding-top:16px}.two-thirds-secondary>p{margin:9px 20px 9px 0}#repo-create{margin-top:-6px}#first-time{padding-top:10px}#repo-import #old-repo{padding-top:6px}#clone-url-https,#clone-url-ssh{margin:0;font-size:12px;line-height:1.5}.diff-too-large{padding:8px}.newform{width:560px}.newform fieldset{margin:0;border:0;padding:0}.newform .checkbox.related-to-preceding{margin-top:4px}.newform label,.newform legend{display:block;padding:16px 0 4px;font-weight:bold;color:#666}.newform fieldset label,.newform .checkbox label{padding:inherit;font-weight:inherit}.newform .required label:after,.newform .required legend:after{content:"\0020(required)";font-weight:normal;font-style:italic;font-size:11px;color:#888}.newform div,.newform fieldset{position:relative;clear:both}.newform fieldset.no-legend,.newform .submit{padding-top:16px}.newform .submit{margin-bottom:20px}.newform input[type="file"],.newform input[type="submit"]{margin:0}.newform .left,.newform .right{width:270px}.newform .left{clear:left;float:left;margin-right:0}.newform .right{clear:right;float:right;margin-left:0}.newform input[type="url"],.newform input[type="text"],.newform input[type="password"],.newform input[type="email"]{margin:0;width:550px;font-size:16px;line-height:20px;color:#222}.newform input[type="url"],.newform input[type="text"],.newform input[type="password"],.newform input[type="email"],textarea{border:1px solid #ccc;border-bottom-color:#999;padding:4px}.newform input[type="url"]:focus,.newform input[type="text"]:focus,.newform input[type="password"]:focus,.newform input[type="email"]:focus,textarea:focus{-moz-box-shadow:1px 1px 3px #6ae,1px -1px 3px #6ae,-1px 1px 3px #6ae,-1px -1px 3px #6ae}.newform input[type="url"][disabled],.newform input[type="text"][disabled],.newform input[type="password"][disabled],.newform input[type="email"][disabled],.newform input[type="url"][readonly],.newform input[type="text"][readonly],.newform input[type="password"][readonly],.newform input[type="email"][readonly]{background-color:#f8f8f8;cursor:default}.newform .left input[type="url"],.newform .left input[type="text"],.newform .left input[type="password"],.newform .left input[type="email"],.newform .right input[type="url"],.newform .right input[type="text"],.newform .right input[type="password"],.newform .right input[type="email"]{width:260px}.newform textarea,.ui-dialog .newform textarea{margin:0;width:550px;padding:4px;font:16px/20px Helvetica,Arial,sans-serif;color:#222}.newform select{margin:0;width:270px}.newform option{font-size:13px}.newform legend.singleton{visibility:hidden}.newform .checkbox legend,.newform .radio legend{padding-bottom:6px}.newform .checkbox label,.newform .radio label{margin:4px 0 4px 20px;color:#555}.newform .checkbox input,.newform .radio input{position:absolute;left:0;top:1px;margin:0}.newform .checkbox input{top:5px}.newform h1,.newform h2,.newform h3,.newform h4,.newform h5,.newform h6{clear:both;padding-top:20px;font-size:16px;line-height:20px}.newform>.message_{margin:10px 0;border:1px solid #ccc;padding:10px 0 10px 16px}.newform>.message_ h4{margin:0;padding:4px 0 4px 36px;line-height:19px;font-size:14px}.newform .field_.message_{border:1px solid #ccc;padding:5px 0 5px 36px;margin:5px 0;list-style:disc}.tooltip_.message_.note_>*:first-child,.newform .field_.message_>*:first-child{margin:0 0 0 -30px;background-repeat:no-repeat;background-position:0 0;padding:4px 0 4px 30px;list-style:none;font-weight:bold}.newform .field_.message_.error_>*:first-child{background-image:url(../img/icons/atl/red-warning-icon.png);color:#b50000}.newform .field_.message_.warning_>*:first-child{background-image:url(../img/icons/atl/yellow-warning-icon.png);background-position:0 2px}.newform .field_.message_.success_>*:first-child{background-image:url(../img/icons/atl/green-upgrade-arrow-icon.png);color:#2d6b00}.tooltip_.message_.note_>*:first-child,.newform .field_.message_.note_>*:first-child{background-image:url(../img/icons/info.png);background-position:6px 4px}.newform .description{font-size:11px}.newform .checkbox label,.newform .radio label{display:block;padding-left:20px;text-indent:-20px;margin-left:0}.newform .checkbox label input,.newform .radio label input{width:13px;height:13px;padding:0;margin:0 3px 0 0;vertical-align:bottom;position:relative;top:-1px;*overflow:hidden}.newform .action{clear:both;float:left;margin:10px 0 0}.newform .action input[type=submit]{margin:0 0 10px}.newform.repo-create #id_description{height:100px}#repo-import .source-fields{display:none}#repo-import #id_url{width:534px;padding-right:20px}#repo-import #id_url.spinner,#repo-import #id_url.success,#repo-import #id_url.warning,#repo-import #id_url.error{background-image:url(../img/loading.gif);background-position:532px 5px;background-repeat:no-repeat}#repo-import #id_url.success{background-image:url(../img/icons/fugue/tick.png)}#repo-import #id_url.warning{background-image:url(../img/icons/fugue/exclamation.png)}#repo-import #id_url.error{background-image:url(../img/icons/fugue/slash.png)}#repo-import #url-auth-fields{display:none}.newform.repo-create #id_name{background:url(../img/icons/fugue/lock.png) no-repeat scroll 249px 50%;padding-right:20px;width:244px}.newform.repo-create #id_name.public{background-image:url(../img/icons/fugue/lock_disable_unlock.png)}.newform .warning_{width:275px;float:left;clear:both;margin:0;padding:0}.newform .warning_ p{margin:15px 0 0;padding:0 0 0 35px;color:#333;background:url("../img/icons/atl/warning-icon.png") 0 5px no-repeat}.newform .warning_{width:275px;float:left;clear:both;margin:0;padding:0}.newform .warning_ p{margin:15px 0 0;padding:0 0 0 35px;color:#333;background:url("../img/icons/atl/warning-icon.png") 0 5px no-repeat}.page-buttons{margin:0 0 20px 0;overflow:auto}.page-buttons-left,.page-buttons-right{border:1px solid #adf;font-size:18px;text-align:center;outline:0}.page-buttons-left{float:left;-webkit-border-radius:10px 0 0 10px;-moz-border-radius:10px 0 0 10px;border-radius:10px 0 0 10px;border-right:0;width:464px}.page-buttons-right{float:right;-webkit-border-radius:0 10px 10px 0;-moz-border-radius:0 10px 10px 0;border-radius:0 10px 10px 0;width:463px}.page-buttons-selected,.page-buttons-unselected:hover{background:#147;background:-webkit-gradient(linear,left top,left bottom,from(#147),to(#036));background:-webkit-linear-gradient(#147,#036);background:-moz-linear-gradient(top,#147,#036);background:-o-linear-gradient(#147,#036);border-color:#036}.page-buttons-unselected{background:#f2faff;background:-webkit-gradient(linear,left top,left bottom,from(#fafaff),to(#eafaff));background:-webkit-linear-gradient(#fafaff,#eafaff);background:-moz-linear-gradient(top,#fafaff,#eafaff);background:-o-linear-gradient(#fafaff,#eafaff)}.page-buttons a,.page-buttons span{display:block;text-decoration:none}.page-buttons-selected{cursor:default}.page-buttons-link{padding:16px 10px 0 20px}.page-buttons-unselected:hover .page-buttons-link,.page-buttons-selected .page-buttons-link{color:white}.page-buttons-description{padding:2px 10px 12px 20px;font-size:12px;color:#999}.bb-dropdown{position:relative;height:28px;background-color:#cde;background-color:rgba(187,204,221,0.75);background-image:url();background-repeat:no-repeat;background-position:100%;padding:0 28px 0 0;cursor:default}.bb-dropdown.singleton{padding:0}.bb-dropdown:hover{background-color:#bcd}.bb-dropdown ::-moz-selection{background-color:transparent}.bb-dropdown ::selection{background-color:transparent}.bb-dropdown,.bb-dropdown>ul{-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;border:2px solid #bcd}.bb-dropdown>ul{position:relative;margin:-2px;background:#fff;line-height:25px;z-index:1}.bb-dropdown>ul>li{display:none;margin:1px;height:24px;border:1px solid #ddd;padding:0 10px;overflow:hidden}.bb-dropdown>ul>li.selected{display:block;border-color:#bcd}.bb-dropdown.open>ul>li{display:block}.bb-dropdown.open>ul>li.selected{background:#fffbe1!important}.bb-dropdown.open>ul>li:hover{background:#fff6c2!important}.bb-button{display:inline;margin:10px auto;-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px;border:0;background:#147;background:-webkit-gradient(linear,0 0,0 100%,from(#147),to(#036));background:-webkit-linear-gradient(#147,#036);background:-moz-linear-gradient(top,#147,#036);background:-o-linear-gradient(#147,#036);padding:5px 10px;font-size:14px;text-shadow:1px 1px 0 #333;color:#fff;cursor:pointer}.bb-button:hover{background:#036;background:-webkit-gradient(linear,0 0,0 100%,from(#036),to(#025));background:-webkit-linear-gradient(#036,#025);background:-moz-linear-gradient(top,#036,#025);background:-o-linear-gradient(#036,#025)}.bb-button:active{background:#025;background:-webkit-gradient(linear,0 0,0 100%,from(#025),to(#036));background:-webkit-linear-gradient(#025,#036);background:-moz-linear-gradient(top,#025,#036);background:-o-linear-gradient(#025,#036)}.bb-button.accept{background:#282;background:-webkit-gradient(linear,0 0,0 100%,from(#282),to(#171));background:-webkit-linear-gradient(#282,#171);background:-moz-linear-gradient(top,#282,#171);background:-o-linear-gradient(#282,#171)}.bb-button.accept:hover{background:#171;background:-webkit-gradient(linear,0 0,0 100%,from(#171),to(#060));background:-webkit-linear-gradient(#171,#060);background:-moz-linear-gradient(top,#171,#060);background:-o-linear-gradient(#171,#060)}.bb-button.accept:active{background:#060;background:-webkit-gradient(linear,0 0,0 100%,from(#060),to(#171));background:-webkit-linear-gradient(#060,#171);background:-moz-linear-gradient(top,#060,#171);background:-o-linear-gradient(#060,#171)}.bb-button.reject{background:#c00;background:-webkit-gradient(linear,0 0,0 100%,from(#c00),to(#b00));background:-webkit-linear-gradient(#c00,#b00);background:-moz-linear-gradient(top,#c00,#b00);background:-o-linear-gradient(#c00,#b00)}.bb-button.reject:hover{background:#b00;background:-webkit-gradient(linear,0 0,0 100%,from(#b00),to(#a00));background:-webkit-linear-gradient(#b00,#a00);background:-moz-linear-gradient(top,#b00,#a00);background:-o-linear-gradient(#b00,#a00)}.bb-button.reject:active{background:#a00;background:-webkit-gradient(linear,0 0,0 100%,from(#a00),to(#b00));background:-webkit-linear-gradient(#a00,#b00);background:-moz-linear-gradient(top,#a00,#b00);background:-o-linear-gradient(#a00,#b00)}#svn-import{padding:0 25px}#svn-import>#masking-box{margin:0 -25px}#svn-import>#masking-box *:hover{cursor:default}#svn-import>#masking-box a:hover{text-decoration:none}#svn-import>#masking-box #toggle-repo-content{background-color:#eee}#svn-import>#masking-box>#mask{position:absolute;left:-2px;top:0;right:-2px;bottom:0;filter:progid:DXImageTransform.Microsoft.gradient(startColorStr=#80FFFFFF,endColorStr=#80FFFFFF);background:rgba(255,255,255,0.5);z-index:99}#svn-import>h1{margin-top:20px}#svn-import>p{margin:1em 0;font-size:14px;line-height:1.5}#svn-import #progress{margin:10px 0;height:24px;padding:1px}#svn-import.error-state h1+p,#svn-import.error-state #progress,#svn-import.warning-state #progress,#svn-import.success-state #progress{display:none}#svn-import #progress,#svn-import #progress>.ui-progressbar-value{-webkit-border-radius:0;-moz-border-radius:0;border-radius:0}#svn-import #progress>.ui-progressbar-value{margin:0;background:#36b;background:-webkit-gradient(linear,0 0,0 100%,from(#69e),to(#36b));background:-webkit-linear-gradient(#69e,#36b);background:-moz-linear-gradient(top,#69e,#36b);background:-o-linear-gradient(#69e,#36b)}#svn-import #progress>.ui-progressbar-value.static-progress{background:url(../img/progressbar.gif) repeat}#svn-import #progress>.ui-progressbar-value>span{position:absolute;top:0;left:0;bottom:0;right:0;display:block;background:-webkit-gradient(linear,0 0,100% 100%,from(rgba(255,255,255,0.15)),color-stop(0.25,rgba(255,255,255,0.15)),color-stop(0.25,transparent),color-stop(0.5,transparent),color-stop(0.5,rgba(255,255,255,0.15)),color-stop(0.75,rgba(255,255,255,0.15)),color-stop(0.75,transparent),to(transparent));background:-moz-linear-gradient(-45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background:-o-linear-gradient(-45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);-webkit-background-size:38px 38px;-moz-background-size:38px 38px;background-size:38px 38px;padding:0 10px;font-weight:bold;font-size:16px;line-height:26px;color:#999;cursor:default}#svn-import #progress>.ui-progressbar-value>span::-moz-selection{background:transparent}#svn-import #progress>.ui-progressbar-value>span::selection{background:transparent}#svn-import h2{position:absolute;left:-9999px}#svn-import #log{margin:10px 0;border:solid 1px #e7e7e7;padding:1px}#svn-import #log>ol{height:288px;background:#000;padding:4px 8px;line-height:18px;font-family:"Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace;color:#fff;overflow:auto}#svn-import #log>ol ::-moz-selection{background:rgba(187,187,187,0.5)}#svn-import #log>ol ::selection{background:#bbb}#svn-import #log time{color:#6ff}#svn-import #status{display:none;margin:10px 0 0;width:876px;border:1px solid #e7e7e7;background:0;padding:1px;font-weight:bold}#svn-import.error-state #status,#svn-import.warning-state #status,#svn-import.success-state #status{display:block}#svn-import #status>p{border-width:1px;border-style:solid;background-repeat:no-repeat;background-position:10px 7px;padding:0;text-align:center;font-size:14px;line-height:22px;text-transform:uppercase}#svn-import.error-state #status>p{border-color:#b50000;background:#ffeae8;color:#b50000}#svn-import.warning-state #status>p{border-color:#fed;background-color:#fff6c2;color:#666}#svn-import.success-state #status>p{border-color:#480;background:#e7f9e0;color:#2d6b00}#consumers table{margin-top:10px;border-spacing:0}#consumers th{border-bottom:2px solid #ddd;padding:6px 0 6px 24px;font-weight:bold;color:#666}#consumers th.creation-date{width:100px}#consumers th.revoke-access{text-indent:-9999px;overflow:hidden}#consumers table td{border-bottom:1px solid #ddd;padding:6px 0 6px 24px}#consumers th.consumer,#consumers td.consumer{width:180px;padding-left:0;font-weight:bold}#consumers th.description,#consumers td.description{width:558px}#consumers td.creation-date{width:100px;color:#666}#consumers td.revoke-access>a{position:relative;top:-2px;display:block;margin-bottom:-20px;width:20px;height:20px;background:url(../img/icons/fugue/cross-sprite.png) no-repeat;text-indent:-9999px;overflow:hidden}#consumers td.revoke-access>a:focus,#consumers td.revoke-access>a:hover{background-position:0 -20px}#changeset h1{margin:0;font-weight:normal;font-size:18px;line-height:2;color:#666}#changeset h1>span>a{font-size:16px;color:#aaa}#changeset #compare h3>a{color:inherit}#changeset #compare h3>a:hover{color:#333}#changesets h1{margin:0;font-weight:normal;font-size:18px;line-height:2;color:#666}#changesets h1>span{color:#aaa}#changesets #commit-filter-help{display:none;padding:0 0 30px}#changesets #commit-filter-help h2{margin:0;padding:0;line-height:2}#changesets #commit-filter-help dl{border-bottom:1px solid #ddd;overflow:auto}#changesets #commit-filter-help dt,#changesets #commit-filter-help dd{border-top:1px solid #ddd;padding:6px 0;line-height:18px}#changesets #commit-filter-help dt{display:block;clear:both;float:left;width:300px;font-weight:bold;font-family:monospace}#changesets #commit-filter-help dd{display:block;float:right;width:630px}#changesets #commit-filter-help dl p{margin:0}#changesets #commit-filter-help dl p+p{margin-top:9px}#changesets #commit-filter-help code{border:1px solid #ddd;background:#eee;padding:0 2px;font-family:monospace;line-height:1}#changesets #shortlogs-changes .newtable td.text a{color:inherit}#changesets>#commit-history{clear:both}#changesets #commit-history .newtable{width:100%}#changesets #commit-history .newtable .hash{width:95px;padding-right:0}#changesets #commit-history .newtable .text{width:490px}#changesets #commit-history .newtable .comments{width:25px;padding-right:0}#changesets #commit-history .newtable .comments a{position:relative;top:3px}#changesets #commit-history .newtable .comments img{border:0;margin:0;float:none}#changesets #commit-history .newtable .comments span{display:none}#changesets #commit-history .newtable th.comments{text-indent:-9999px}#changesets #shortlogs-changes{position:relative}#changesets #changesets-graph{position:absolute;top:-40px;left:-400px;width:400px}#changesets #changesets-inner{position:static;margin:0 0 10px}#changesets #changesets-inner h3{margin:-1px 0 0;border:1px solid #ddd;background:#eee;padding:10px 24px;font-size:15px}#changesets .paginator{margin-top:9px}#changesets #changesets-inner>ol{font:12px/1.5em "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace}#changesets #changesets-inner>ol>li{margin:-1px 0 0;border:1px solid #ddd;padding:10px 300px 10px 24px}#changesets #changesets-inner>ol>li:nth-child(even){background:#fcfcfc}#changesets #changesets-inner h4{position:absolute;left:-9999px}#changesets #changesets-inner p{margin:.25em 0 .5em;width:600px;line-height:inherit}#changesets #changesets-inner .metadata{position:relative;margin:.5em 0 0;min-height:34px;padding:2px 0 2px 45px;font-family:Helvetica,Arial,sans-serif}#changesets #changesets-inner .metadata .avatar{position:absolute;left:-3px;top:0}#changesets #changesets-inner .metadata dt{position:absolute;left:-9999px}#changesets #changesets-inner .metadata time,#changesets #changesets-inner .metadata dd[title]{color:#747474}#changesets #changesets-inner li{position:relative}#changesets #changesets-inner .metadata+p{position:absolute;right:220px;bottom:12px;display:block;width:auto}#changesets #commit-history #file-history .hash{width:120px}#changesets #commit-history #file-history .added,#changesets #commit-history #file-history .removed{width:26px;background:0;padding-right:4px;text-align:right;color:#282}#changesets #commit-history #file-history .removed{color:#c33}#changesets #commit-history #file-history th.added,#changesets #commit-history #file-history th.removed{background:url(../img/icons/green/add.gif) no-repeat 100%;line-height:999px;overflow:hidden}#changesets #commit-history #file-history th.removed{background-image:url(../img/icons/red/remove.gif)}#changesets #commit-history #file-history .text{width:410px}#changesets #commit-history #file-history.no-diffstat .text{width:470px}#changesets #commit-history #file-history .text a{color:inherit}#changesets #commit-history #file-history td.labels{right:210px}#changesets #commit-history #file-history td.branch{right:140px}#changesets #commit-history #file-history td.branch>span{float:right;margin-left:4px}#changesets #commit-history #file-history td.branch>span.tag{background:#fff url(../img/icons/fugue/tags.png) no-repeat 4px;padding-left:24px}#changesets #commit-history #file-history .date{padding-left:20px}#changesets #commit-history #file-history .link{width:60px;padding-right:0}#changesets #commit-history #file-history th.link{margin-top:-6px;padding-bottom:6px;line-height:1}#commit-filter-control{float:right;margin-top:4px;-webkit-border-radius:13px;-moz-border-radius:13px;border-radius:13px;border:1px solid #ccc;background:#eee;background:-webkit-gradient(linear,0 0,0 100%,from(#fff),to(#ddd));background:-webkit-linear-gradient(#fff,#ddd);background:-moz-linear-gradient(top,#fff,#ddd);background:-o-linear-gradient(#fff,#ddd);padding:0 4px 0 0;position:relative;z-index:1}#commit-filter-control:before{position:absolute;left:-9999px;top:5px;display:block;width:16px;height:16px;background:url(../img/loading.gif);content:""}#commit-filter-control.loading:before{left:-24px}#commit-filter-control div,#commit-filter-control ul{float:left}#commit-filter-control li{float:left;width:23px;height:26px;text-indent:-9999px}#commit-filter-control a{display:block;width:100%;height:100%;border-left:1px solid #ccc;background:no-repeat 50%}#commit-filter-control a[href="#help"]{border-left:none;background-image:url()}#commit-filter-control a[rel="prev"]{background-image:url()}#commit-filter-control.no-help a[rel="prev"]{border-left:none}#commit-filter-control a[rel="prev"][href]{background-image:url()}#commit-filter-control a[rel="next"]{background-image:url()}#commit-filter-control a[rel="next"][href]{background-image:url()}#commit-filter-control a[rel="last"]{background-image:url()}#commit-filter-control a[rel="last"][href]{background-image:url()}#commit-filter-control input{margin:2px;width:300px;font-size:16px;border:1px solid #cfcfcf;background:#fff url(../images/search-icon.png) no-repeat 4px;border-radius:12px}body.non-webkit #commit-filter-control input{font-size:13px;padding:3px 0 3px 23px}#compare #interdiff-heading{font-weight:normal;font-size:18px;color:#999}#compare h2,#compare h3{margin:0;padding-top:10px;font-weight:normal;font-size:18px;line-height:2;color:#666}#compare h2>span{color:#aaa}#compare h3>a{color:inherit}#compare #repo-heading>a{color:#2b547d}#compare .incoming-outgoing{position:absolute;right:0;margin-top:-30px}#compare>p,#compare #compare-header>p{margin:.5em 0 1em}#compare>pre,#compare #compare-header>pre{margin:1em 0;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;border:1px solid #ddd;background:#eee;padding:.5em 1em;font-size:12px;line-height:16px}#compare .newtable .hash{width:100px}#compare .newtable .text{width:490px}#compare .newtable .date{width:100px}#compare .newtable .file{width:778px;background-repeat:no-repeat;background-position:2px}#compare .newtable td.file{width:750px;padding-left:28px}#compare .newtable td.file strong.conflicts{color:#c00;font-weight:normal}#compare .newtable .file-added{background-image:url(../img/icons/green/add.gif)}#compare .newtable .file-modified{background-image:url(../img/icons/orange/edit.gif)}#compare .newtable .file-removed{background-image:url(../img/icons/red/remove.gif)}#compare .newtable .file-conflicts{background-image:url(../img/icons/red/warning.gif)}#compare .newtable .lines-added,#compare .newtable .lines-removed{width:36px;padding:0;text-align:center}#compare .newtable th.lines-added,#compare .newtable th.lines-removed{background:url(../img/icons/green/add.gif) no-repeat 10px;text-indent:-9999px}#compare .newtable th.lines-removed{background-image:url(../img/icons/red/remove.gif);padding-right:60px}#compare .newtable td.lines-added{color:#282}#compare .newtable td.lines-removed{padding-right:60px;color:#c33}#compare a[href="#files"]{position:absolute;right:0;margin-top:20px;width:15px;height:15px;background:url(../img/up-arrow-sprite.png);text-indent:-9999px}#compare a[href="#files"]:focus,#compare a[href="#files"]:hover{background-position:-15px}#compare a[href="#files"]~a[href="#files"]{margin-top:30px}#compare #changesets-graph{position:absolute;top:-40px;left:-400px;width:400px}#compare #shortlogs-changes .newtable td.text a{color:inherit}#compare #shortlogs-changes .newtable{width:100%}#compare #shortlogs-changes .newtable .hash{width:95px;padding-right:0}#compare #shortlogs-changes .newtable .text{width:490px}#compare #shortlogs-changes .newtable .comments{width:25px;padding-right:0}#compare #shortlogs-changes .newtable .comments a{position:relative;top:3px}#compare #shortlogs-changes .newtable .comments img{border:0;margin:0;float:none}#compare #shortlogs-changes .newtable .comments span{display:none}#compare #shortlogs-changes .newtable th.comments{text-indent:-9999px}#compare #shortlogs-changes{position:relative}#cross-changeset-file-diff h1{padding:16px 0 6px;font-weight:normal;font-size:18px;line-height:24px;color:#666}#cross-changeset-file-diff h1>a{color:#2b547d}#cross-changeset-file-diff .newtable .file{width:778px}#cross-changeset-file-diff .newtable .lines-added,#cross-changeset-file-diff .newtable .lines-removed{width:36px;padding:0;text-align:center}#cross-changeset-file-diff .newtable th.lines-added,#cross-changeset-file-diff .newtable th.lines-removed{background:url(../img/icons/green/add.gif) no-repeat 10px;text-indent:-9999px}#cross-changeset-file-diff .newtable th.lines-removed{background-image:url(../img/icons/red/remove.gif);padding-right:60px}#cross-changeset-file-diff .newtable td{background-repeat:no-repeat;background-position:2px}#cross-changeset-file-diff .newtable td.file{width:750px;padding-left:28px}#cross-changeset-file-diff .newtable td.lines-added{color:#282}#cross-changeset-file-diff .newtable td.lines-removed{padding-right:60px;color:#c33}#cross-changeset-file-diff .newtable .file-added{background-image:url(../img/icons/green/add.gif)}#cross-changeset-file-diff .newtable .file-modified{background-image:url(../img/icons/orange/edit.gif)}#cross-changeset-file-diff .newtable .file-removed{background-image:url(../img/icons/red/remove.gif)}#cross-changeset-file-diff #changeset-diff h3,#cross-changeset-file-diff #changeset-diff a[href="#files"]{display:none}.user-comments{clear:both}.user-comments #comments-list{padding:0}.user-comments #comments-list:not(:empty){margin-bottom:20px}.user-comments #comments-list>li{margin:0}.user-comments #comments-list>li:before{content:""}.user-comments #comment-header{margin:6px 0;padding:10px 0 0;font-weight:normal;font-size:18px;line-height:24px;color:#666}.user-comments .comment-not-spam-link{display:none;position:absolute;right:40px;bottom:10px}.user-comments .user-comment .comment-not-spam-link a span{background:url(../img/icons/fugue/plus_small_circle.png) no-repeat left 65%;padding:2px 0 2px 14px;font-size:90%}.user-comments .user-comment.is-spam{background:#eee url(../img/diagonal_transparent_bg.png) repeat left top}.user-comments .comment-permalink{position:absolute;right:16px;bottom:10px}.user-comments .comment-permalink a span{background:url(../img/icons/permalink_light_16.png) no-repeat 50% 50%;width:16px;height:16px;display:block;text-indent:-9999px;cursor:pointer;margin-top:1px}.user-comments .comment-permalink a:hover span{background-image:url(../img/icons/permalink_dark_16.png)}.user-comments h4{margin-top:0}.user-comments blockquote,.user-comments pre{margin:9px 0;border:1px solid #ddd;padding:9px 12px}.user-comments blockquote{background:#fff;padding-top:0;padding-bottom:0}.user-comments textarea{font:12px/1.5 "Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace}.user-comments .user-comment{position:relative;margin:-1px 0 0;border-bottom:1px solid #ddd;border-top:1px solid #ddd;background:#fff;padding:10px 16px 10px 58px;font-size:13px}.user-comments .deleted{position:relative;margin:-1px 0 0;border-bottom:1px solid #ddd;border-top:1px solid #ddd;background:#fff;padding:10px 16px 10px 5px}.user-comments .user-comment img.avatar{width:32px;height:32px}#changeset .user-comment p{margin:9px 0;padding:0}.user-comments .user-comment dt{position:absolute;text-indent:-9999px}.user-comments .username{display:inline;font-weight:bold;color:#2b547d}.user-comments .user-comment .newform{width:auto}.user-comments .user-comment .changeset{display:none}.user-comments .user-comment .date{position:absolute;right:16px;top:10px;color:#999;font-size:90%}.user-comments .user-comment .comment-actions{position:relative;height:14px;border:0;padding:0;line-height:1;color:#666}.user-comments .user-comment .comment-actions a{font-size:90%}.user-comments .user-comment .comment-actions>li{float:left;margin:0}.user-comments .user-comment .comment-actions>li+li{margin-left:8px;border-left:1px solid #999;padding-left:8px}.user-comments .user-comment .comment-actions>li:before{content:""}.user-comments .child-comments{padding:0}.user-comments .child-comments>li{margin:0}.user-comments .child-comments>li:before{content:""}.user-comments .child-comments .user-comment{margin-left:48px}.user-comments .child-comments .child-comments .user-comment{margin-left:96px}.user-comments .child-comments .deleted{margin-left:48px}.user-comments dd.comment{margin:-5px 0 0}.user-comments .comment-reply{display:none;padding-top:10px}.user-comments #comment-reply-template{display:none}.user-comments ul .comment-reply{margin-left:58px}.user-comments ul .child-comments .child-comments .comment-reply{margin-left:106px}.user-comments .comment-reply .username{position:relative;padding:0 12px 0 48px}.user-comments .comment-reply img.avatar{position:absolute;left:0;top:10px}.user-comments .comment-reply .comment-form-container{margin-left:48px}.user-comments .comment-reply .comment-form-container .newform{width:100%}.user-comments .comment-reply .comment-form-container .newform .textarea-container{padding:0 0 5px}.user-comments .comment-reply .comment-form-container .newform textarea{margin:0;width:100%;height:108px;padding:3px 6px}.user-comments .comment-reply .comment-form-container .newform .submit{padding-top:0}.user-comments .new-comment-link{display:block;margin:10px 0}.composing-comment .user-comments .new-comment-link{display:none}.user-comments .markItUpContainer{margin-right:14px}.user-comments .comment-edit{display:none}.user-comments .user-comment .newform{position:relative;width:100%;padding-right:15px}.user-comments .comment-edit textarea{position:relative;margin:0;height:108px;width:100%}#dashboard #newsfeed h2{margin-bottom:11px}#dashboard .detailed p{width:600px}#dashboard #relevant-repos{float:right;width:300px;font-size:14px;line-height:18px}#dashboard #relevant-repos h2{float:left;margin-bottom:4px;font-size:16px;line-height:20px;color:#2b547d}#dashboard #relevant-repos .create-repo,#dashboard #relevant-repos #following-only{position:relative;top:2px;display:inline-block;margin:0 -4px -10px 8px;width:16px;height:16px;background:url(../img/icons/fugue/plus.png);text-indent:-9999px}#dashboard #relevant-repos #following-only{display:none!important}#dashboard #relevant-repos #following-only,#dashboard #relevant-repos.following-only #following-only:hover{background-image:url(../img/icons/fugue/heart_sprite.png);background-position:0 -96px}#dashboard #relevant-repos #following-only:hover,#dashboard #relevant-repos.following-only #following-only{background-position:0 -32px}#dashboard #relevant-repos ol{margin-top:4px}#dashboard #relevant-repos>ol>li{margin-top:13px}#dashboard #relevant-repos h3{clear:both;margin:0;font-size:14px;color:#747474}#dashboard #relevant-repos h3 a{color:inherit}#dashboard #relevant-repos li li{position:relative;background:url(../img/icons/fugue/lock_disable_unlock.png) no-repeat 22px 4px;padding:4px 0 4px 44px}#dashboard #relevant-repos li.private{background-image:url(../img/icons/fugue/lock.png)}#dashboard #relevant-repos .fork{display:inline-block;width:16px;background:url(../img/icons/fork_right.png) no-repeat 50%;text-indent:-9999px}#dashboard #relevant-repos .follow{position:absolute;left:0;top:5px;display:block;width:16px;height:16px;background:url(../img/icons/fugue/heart_sprite.png) no-repeat;padding:0;text-indent:-9999px;overflow:hidden}#dashboard #relevant-repos .follow:hover,#dashboard #relevant-repos .follow.following{background-position:0 -32px}#dashboard #relevant-repos .follow.following:hover{background-position:0 -64px}#dashboard #relevant-repos.following-only li li{display:none}#dashboard #relevant-repos.following-only li.following{display:inherit}#repobox a.user-repos{display:block}#repobox li.selected{margin-left:5px}#repobox li.selected a{text-decoration:underline;color:#2b547d}#repobox li.selected a:after{content:'\00A0\2192'}#repobox .unfolded:before{content:"\25BC\00A0"}#repobox .folded:before{content:"\25B6\00A0"}#dashboard #repobox{padding:8px;width:250px;background:rgba(249,249,249,1.0);overflow:auto}#dashboard #repobox #repolist{clear:both}#repobox #repo-actions{border:1px solid #ddd;background:#eee;display:block;height:24px;padding:2px;margin:6px 0 6px 0}#repo-actions input{float:right;width:98%;margin-right:1%}#descendants{padding:0 25px}#descendants #tabs,#descendants #repo-menu,#descendants #repo-desc{margin-left:-25px;margin-right:-25px}#descendants h2{margin-top:25px;color:#333}#descendants ol.detailed{margin:0 -10px}#descendants ol.detailed dd.compare a{display:block;float:left;margin:2px 0 0 4px;width:20px;height:20px;background:url(../img/arrow_switch.png) no-repeat 50%;text-indent:-9999px}#descendants ol.detailed dd.avatar{border:1px solid #e7e7e7;padding:2px}#descendants ol.detailed dt.url{position:inherit;left:inherit;clear:left;float:left}#descendants ol.detailed dt.url:after{content:":"}#descendants ol.detailed dt.url+dd{float:left;margin:0 4px}#descendants ol.detailed dd.size{float:left;color:#999}#descendants ol.detailed dd.size:before{content:"("}#descendants ol.detailed dd.size:after{content:")"}#descendants ol.detailed p{width:600px}#downloads #uploaded-files .name{width:550px}#downloads #uploaded-files .count{width:78px}#downloads #uploaded-files .delete{width:22px;background:url(../img/loading.gif) no-repeat -9999px;padding-right:0}#downloads #uploaded-files .delete>a{display:block;margin:6px 0;width:16px;height:16px;background:url(../img/icons/fugue/minus_circle.png);text-indent:-9999px;overflow:hidden}#downloads #uploaded-files .delete>a.deleting{background:url(../img/loading.gif)}#downloads .name{width:220px}#downloads td.name{font-family:Courier,"Courier New",monospace}#downloads .text{width:310px}#downloads .text>a{color:inherit}#downloads .size,#downloads .hash,#downloads .type{width:100px}#downloads .type ul{list-style:none}#downloads .type li{display:inline}#downloads .type li:after{content:","}#downloads .type li+li+li:after{content:""}#downloads #new-download label{position:absolute;left:-9999px}#downloads #new-download .file{padding-top:5px}#downloads #new-download .submit{display:none}#explore #explore-h1 h1{float:left}#explore #explore-h1:after{display:block;clear:both;height:0;content:".";visibility:hidden}#explore #explore-h1 p{position:relative;left:5px;top:7px;color:#777}#explore #tweet-buzz{float:right;margin-top:15px;width:31%}#explore #tweet-buzz-loading{color:#999}#explore #tweet-buzz-results{margin-top:11px}#explore #tweet-buzz-results p{margin:0 0 10px 46px;font-size:12px}#explore #tweet-buzz-results p.tweet a{word-wrap:break-word}#explore #tweet-buzz-results a.img{float:left}#explore #tweet-buzz-results p.name{margin-bottom:2px;font-size:15px}#explore #tweet-buzz-results a.time{font-size:12px;color:#999}#explore .short-repo-list{position:relative;margin:15px 0 25px;width:66%}#explore .short-repo-list .detailed{margin-bottom:3px}#explore .short-repo-list .detailed>li{margin-left:-8px}#explore .short-repo-list h2,#explore #tweet-buzz h2{font-size:16px;color:#555}#explore .short-repo-list .week .info-button-label{width:32px}#explore .short-repo-list .week dd.commits{right:201px}#explore .short-repo-list dd.public.repository{display:none}#explore .trending-repo-group.month,#explore .trending-repo-group.all{display:none}#explore ul.trending-periods{position:absolute;right:10px;top:0}#explore ul.trending-periods li{display:inline;padding:0 2px}#explore ul.trending-periods li.selected{font-weight:bold}#explore #search-form{float:left;margin-bottom:10px}#explore #search-form #search-text{display:block;float:left;margin:0;width:400px;height:20px;padding:4px 5px 4px 30px;font-size:18px;line-height:20px;border:1px solid #ddd;background:url(../images/search-docs-bg-30px.png) no-repeat 0 -1px}#explore .searchInput .selectors-wrap{clear:both;float:left;margin-right:.5em}.actions-wrap{float:left;margin-left:.5em}#explore #search-form button{display:block;float:left;margin-left:3px;height:30px;border:1px solid #ddd;padding:0 10px;font-size:15px}#explore #search-form+p{clear:both}#explore #sort-repos{float:right;padding-top:3px;font-weight:bold}#explore #sort-repos ul{float:left;margin-left:9px}#explore #sort-repos h4{float:left;margin:0 6px 0 0;width:50px;font-weight:bold;font-size:13px;line-height:24px;color:#393939}#explore #repositories{clear:both;margin:0 -10px}#explore #repositories p{width:600px}#explore dd.avatar img{width:32px;height:32px;border:1px solid #e7e7e7;padding:2px}#explore #repositories em{background-color:#faffa6}#explore .snippet{font-family:Monaco,"Courier New","DejaVu Sans Mono","Bitstream Vera Sans Mono",monospace;background-color:ghostWhite;border:1px solid #dedede;padding:.5em;line-height:1.5em;color:#444}#issues #search{position:relative;float:left;z-index:1}#issues #search label{position:absolute;left:-9999px}#issues #search #q{margin-top:1px;width:200px;font-size:16px}body.non-webkit #issues #search #q{margin-top:0;height:14px;border:1px solid #ccc;background:url(../images/search-icon.png) no-repeat 5px;padding:4px 0 4px 24px;font-size:13px}#issues #filters-filters{margin-top:-10px}#issues #filters-filters input[type="text"]{margin-bottom:2px}#issues #filters-filters #addfilter{margin-top:10px}#issues h1{margin-top:20px}#issues h2{font-size:16px;line-height:20px}#issues #issues-new-form select{max-width:250px}#issues #issues-new-form .markItUpContainer{width:442px}#issues #issues-main-form{padding:0 25px}#issues #issues-issue{margin:0 -25px;border:1px solid #ddd;background:#fff url(../img/layout/bg_general.png) repeat-x;padding:10px 24px}#issues #issues-main-form .avatar{position:absolute;left:24px;top:12px}#issues #issues-issue h1{margin:0 0 0 48px}#issues #issues-issue pre{white-space:pre-wrap}#issues .issues-issue-infotable{margin:0 -25px 10px;border:1px solid #ddd;border-top:0;padding:10px 24px}#issues .issues-issue-infotable td{padding-right:6px}#issues .issues-issue-infotable td.attr{font-weight:bold}#issues .issues-issue-infotable td.val{padding-right:90px}#issues #issues-issue-attachments{margin:0 -25px 10px;border:1px solid #ddd;padding:10px 24px}#issues #attachment_link{background:url(../img/icons/fugue/document__plus.png) no-repeat 0;padding:2px 0 2px 20px}#issues #issues-comments>li{position:relative;margin:-1px -25px 0;border:1px solid #ddd;padding:12px 24px 12px 72px}#issues .issues-comment-form{margin:-1px -25px 0;border:1px solid #ddd;padding:10px 24px}#issues .issues-action-form{margin:-1px -25px 0;border:1px solid #ddd;padding:10px 24px}#issues .buttons{margin:-1px -25px 0;border:1px solid #ddd;padding:9px 21px;overflow:auto}#issues .buttons li input{float:left;margin:4px;border-width:1px;border-style:solid;border-top-color:#ddd;border-left-color:#ddd;border-right-color:#bbb;border-bottom-color:#bbb;background:#eee;padding:2px 8px}#issues .buttons li:first-child input{margin:3px;background:#257;padding:3px 10px;color:#fff}#issues .buttons .cancel{margin:3px}#issues .issues-comments-permalink{float:right}#issues #issues-issue-attachments a.file-icon{margin-right:20px;background:url(../img/icons/fugue/document.png) no-repeat left;padding:2px 0 2px 20px}#issues #issues-issue-attachments table{margin-top:10px;border-collapse:collapse}#issues #issues-issue-attachments td{padding-left:0}#issues #issues-issue-state{float:right;margin-left:30px;margin-bottom:10px}#issues #issues-issue-state h3{margin-top:0;margin-bottom:8px;padding-top:0;text-align:right}#issues #issues-issue-state p{text-align:right}#issues .issues-change-indicator{background:url(../img/icons/silver/new.gif) no-repeat left;padding:2px 0 2px 20px}#issues p.issues-nocomments{padding:10px 0 0 20px}#issues .issues-action-form div.changes{float:right;margin-left:30px;width:380px;line-height:30px}#issues .issues-action-form div.changes dt{display:block;float:left;width:75px}#issues .issues-action-form div.changes select{min-width:175px}#issues .issues-comment-form #id_content{width:690px;height:162px}#issues .issues-comment-form #preview{clear:both;width:651px;border:1px solid #e7e7e7;background:#fcfcfc;padding:9px 24px}#issues #issues-new-form #preview{clear:both;width:400px;border:1px solid #ddd;background:#fcfcfc;padding:0 12px 12px 28px}#issues .issues-issue-description{padding-top:10px;padding-bottom:5px;font-size:1em}#issues .issues-issue-description p{margin-top:1em}#issues .issues-issue-description ul{margin:1.33em 0}#issues .issues-issue-description ol{margin:1.33em 0;padding-left:0;list-style-type:decimal;list-style-position:inside}#issues .issues-issue-description ul li>ul{margin-left:13px}#issues .issues-issue-description ol li>ol{margin-left:18px}#issues .issues-issue-description ol ul,#issues .issues-issue-description ul ol,#issues .issues-issue-description ul ul,#issues .issues-issue-description ol ol{margin-top:0;margin-bottom:0}#issues .issues-comment-form h2,#issues .issues-action-form h2{padding:0;line-height:40px}#issues .issues-action-form p{line-height:27px}#issues .issues-action-form select{margin:0 3px}#issues .issues-change-title{width:885px;border-bottom:1px solid #ddd;padding:10px 20px}#issues .issues-change-title span{font-size:1.3em}#issues .issues-change-title input{width:500px}#issues .issue-follow{background:url(../img/icons/fugue/heart__plus.png) no-repeat left;padding:2px 0 2px 19px}#issues .issue-following{padding:2px 0 2px 19px;background:url(../img/icons/fugue/heart.png) no-repeat left}#issues .button-groups{padding:5px 0 10px;overflow:auto}#issues .button-groups+.sane-defaults{margin-top:-52px}#issues .button-groups .button-group{position:relative;float:left;margin:0 10px 0 0;z-index:1}#issues #create-issue{float:right;margin-right:0}#issues #create-issue a.icon:before{left:8px;background:url(../img/icons/fugue/ticket_plus.png)}#issues #create-query a.icon:before{left:8px;background:url(../img/icons/fugue/gear.png)}#issues #issues-filter-criteria{position:relative;top:60px;margin:0}#issues #issues-list{margin:70px 0 10px;border-bottom:1px solid #ddd;line-height:18px}#issues #issues-filter-criteria:empty+#issues-list{margin-top:54px}#issues #issues-list th>a{color:inherit;outline:0}#issues #issues-list[data-sort-by] th>a:after{position:relative;left:3px;top:3px;display:inline-block;height:15px;content:""}#issues #issues-list[data-sort-by="id"] a[href$="sort=-id"]:after,#issues #issues-list[data-sort-by="created_on"] a[href$="sort=-created_on"]:after,#issues #issues-list[data-sort-by="status"] a[href$="sort=-status"]:after,#issues #issues-list[data-sort-by="responsible"] a[href$="sort=-responsible"]:after,#issues #issues-list[data-sort-by="kind"] a[href$="sort=-kind"]:after,#issues #issues-list[data-sort-by="version"] a[href$="sort=-version"]:after,#issues #issues-list[data-sort-by="milestone"] a[href$="sort=-milestone"]:after,#issues #issues-list[data-sort-by="component"] a[href$="sort=-component"]:after,#issues #issues-list[data-sort-by="priority"] a[href$="sort=-priority"]:after{width:15px;background:url(../img/arrows-sprite.png) -5px -30px}#issues #issues-list[data-sort-by="-id"] a[href$="sort=id"]:after,#issues #issues-list[data-sort-by="-created_on"] a[href$="sort=created_on"]:after,#issues #issues-list[data-sort-by="-status"] a[href$="sort=status"]:after,#issues #issues-list[data-sort-by="-responsible"] a[href$="sort=responsible"]:after,#issues #issues-list[data-sort-by="-kind"] a[href$="sort=kind"]:after,#issues #issues-list[data-sort-by="-version"] a[href$="sort=version"]:after,#issues #issues-list[data-sort-by="-milestone"] a[href$="sort=milestone"]:after,#issues #issues-list[data-sort-by="-component"] a[href$="sort=component"]:after,#issues #issues-list[data-sort-by="-priority"] a[href$="sort=priority"]:after{width:15px;background:url(../img/arrows-sprite.png) -5px -5px}#issues #issues-list th,#issues #issues-list td{height:auto;padding-top:5px;padding-bottom:5px}#issues #issues-list th{margin-bottom:-1px}#issues #issues-list td{border-top:1px solid #ddd;border-bottom:0}#issues #issues-list .text{width:516px}#issues #issues-list.optcols_1 .text{width:411px}#issues #issues-list.optcols_2 .text{width:306px}#issues #issues-list.optcols_3 .text{width:201px}#issues #issues-list .type,#issues #issues-list .priority{width:32px;padding-right:0;text-align:center}#issues #issues-list .state,#issues #issues-list .person{width:60px;padding-left:10px;padding-right:10px}#issues #issues-list .person{width:115px}#issues #issues-list .date{width:105px;padding-left:10px;padding-right:0}#issues #issues-list .component,#issues #issues-list .milestone,#issues #issues-list .version{width:95px;padding-left:10px;padding-right:0}#issues #issues-list td.type a,#issues #issues-list td.priority a{display:block;margin:-1px auto;width:20px;height:20px;background:url(../img/issue-types.png) no-repeat 0 0;text-indent:-9999px}#issues #issues-list td.type .enhancement{background-position:0 -20px}#issues #issues-list td.type .proposal{background-position:0 -40px}#issues #issues-list td.type .task{background-position:0 -60px}#issues #issues-list td.priority a{background-image:url(../img/priorities.png)}#issues #issues-list td.priority .critical{background-position:0 -20px}#issues #issues-list td.priority .major{background-position:0 -40px}#issues #issues-list td.priority .minor{background-position:0 -60px}#issues #issues-list td.priority .trivial{background-position:0 -80px}#issues #issues-list td.person a,#issues #issues-list td.component a,#issues #issues-list td.milestone a,#issues #issues-list td.version a{color:inherit}#issues #issues-list td.person .unassigned{color:#999}#issues #issues-list tr[data-state="resolved"] .state>a{color:#171}#issues #issues-list tr[data-state="on hold"] .state>a{color:#888}#issues #issues-list tr[data-state="invalid"] .state>a{color:#c33}#issues #issues-list tr[data-state="duplicate"] .state>a{color:#606}#issues #issues-list tr[data-state="wontfix"] .state>a{color:#c90}#issues #issues-list td[colspan="6"],#issues #issues-list td[colspan="7"],#issues #issues-list td[colspan="8"],#issues #issues-list td[colspan="9"]{width:910px;font-style:italic}#issues #issues-new-form{border:1px solid #ddd;background:#eee;padding:15px 24px}#issues #issues-new-form #id_title{width:430px}#issues #issues-new-form #id_content{margin-left:0;width:430px;height:162px}#issues #issues-new-form h3{margin-top:20px}#issues div.issue-status{clear:both;margin-top:-4px;margin-bottom:-6px;width:100px}#issues #issues-new-form input[type="text"],#issues #issues-issue input[type="text"],#issues #issues-issue select{min-width:200px}#issues #issues-new-responsible{position:absolute;right:0;top:15px;margin-left:30px;width:380px}#issues #issues-new-responsible td{padding:2px 2px 2px 0;text-align:right}#issues #issues-new-responsible th{padding:2px 10px 2px 0}#issues p.issues-issue-reportedby{margin:0 0 0 48px;padding-bottom:0;color:#747474}#issues .issue-changes li{margin:0}#issues .issue-changes li:before{content:"→\0020"}#issues .issue-changes span{background:#dde7ef;font-weight:bold;color:#333}#issues #issues-comments{line-height:18px}#issues #issues-comments .timestamps{color:#747474}#issues #issues-comments .highlighttable{width:auto}#issues .moderation-controls{text-align:right}#issues .moderation-controls li{display:inline}#issues .moderation-controls a{padding:0 0 0 16px;font-size:inherit}#issues .edit-comment-link{display:block;margin:12px 0 0;padding:0 0 0 18px;font-size:inherit;color:inherit}#issues #filters a[href="#remove"]{display:block;width:16px;height:16px;background:url(../img/icons/fugue/minus_circle.png);text-indent:-9999px;overflow:hidden}#messages #notification-controls{height:42px;border:1px solid #ddd;padding:16px 0 0}#messages #notification-controls input[type="checkbox"]{display:block;float:left;margin:7px 20px}#messages #notification-controls .paginator{margin:2px 20px}#messages #notifications{font-size:13px;line-height:16px}#messages #notifications li{position:relative;margin-top:-1px;min-height:36px;border:1px solid #ddd;padding:12px 20px 10px 300px}#messages #notifications li:nth-child(odd){background:#fcfcfc}#messages #notifications a[data-username],#messages #notifications span.deleted-user{position:absolute;left:105px;top:12px}#messages #notifications time{position:absolute;left:105px;top:32px;font-size:11px}#messages #notifications .avatar{position:absolute;left:52px;top:10px}#messages #notifications h4{margin:0 0 4px;font-weight:normal;font-size:inherit}#messages #notifications .unread h4{font-weight:bold}#messages #notifications input[type="checkbox"]{position:absolute;left:20px;top:23px;margin:0}#messages #no-notifications{border:1px solid #ddd;padding:20px;line-height:18px}#messages #inbox{border:1px solid #ddd;padding:20px}#messages #inbox h1{margin:-8px 0 0 52px;border-bottom:0;padding-bottom:8px;text-align:left}#messages #inbox h2{margin:0;padding:0;font-size:inherit;line-height:38px;color:inherit}#messages #inbox .sane-defaults h2{margin:6px 0;padding:10px 0 0;font-size:18px;line-height:24px;color:#666}#messages #inbox p{margin-bottom:8px;line-height:16px}#messages #inbox ol{margin:0 -20px;border-top:1px solid #ddd}#messages #inbox li{position:relative;min-height:38px;border-bottom:1px solid #ddd;padding:10px 20px 10px 72px}#messages #inbox li:nth-child(odd){background:#fcfcfc}#messages #inbox .avatar{position:absolute;left:20px;top:10px}#messages #inbox form{padding-left:52px}#messages #inbox #id_content{margin-left:0}#messages #inbox .primary-button{margin-right:10px}#messages #compose{border:1px solid #ddd;padding:20px}#messages #compose label{display:block;margin:4px 0 8px;padding:6px}#messages #compose input[type="text"],#messages #compose textarea{margin:4px 0 8px;width:400px;border:1px solid #ddd;padding:6px}#messages #compose input[type="text"]:focus,#messages #compose textarea:focus{border-color:#2b547d}#messages #compose .primary-button{margin-right:10px}#overview>#repo-desc{margin-bottom:15px}#overview>h1{position:absolute;left:-9999px;margin:20px 25px;font-weight:normal;font-size:38px;line-height:1;color:#999}#overview>h1>span{color:#2b547d}#overview>h2{margin:0;font-weight:normal;font-size:18px;line-height:2;color:#666}#overview>h2>span>a{font-size:16px;color:#aaa}#overview>#shortlogs-changes{margin-bottom:25px}#overview>#shortlogs-changes>.maintable th:first-child,#overview>#shortlogs-changes>.maintable td:first-child{padding-left:25px}#overview>.readme{margin:30px 0 0;border:0;padding:0}#overview>.readme>.plaintext{margin:0;border:0;background:0;padding:0;white-space:pre-wrap}#profile{position:relative}#profile h1{display:inline-block;margin:0 0 15px 50px;border:0;padding:0;text-align:left;font-weight:normal;font-size:32px;line-height:38px;color:#2b547d}#profile h1 span{color:#999}#profile>.avatar{position:absolute;left:0;top:0;display:block;border:1px solid #e7e7e7;background:#fff;padding:2px}#profile .actions{float:right;padding-top:18px}#profile .actions li{float:left}#profile .actions li+li{margin-left:20px}#profile .actions a{display:block;background:url(../img/icons/fugue/heart_sprite.png) no-repeat 100% 0;padding:0 24px 0 0;text-decoration:none;color:#525252}#profile .actions .follow:hover,#profile .actions .follow.following{background-position:100% -32px}#profile .actions .follow.following:hover{background-position:100% -64px}#profile .actions .send-message{background-image:url(../img/icons/fugue/mail__arrow_sprite.png)}#profile .actions .send-message:hover{background-position:100% -32px}#profile #metadata{border-width:1px 0;border-style:solid;border-color:#ddd;padding:15px 0;overflow:auto}#profile #metadata .userinfo{float:left;width:380px;line-height:20px}#profile #metadata .userinfo dt{position:relative;left:0;float:left;width:100px;color:#999}#profile #metadata #people{width:430px;float:right;overflow:auto}#profile #metadata #people>div{float:left;width:206px}#profile #metadata #people>div+div{float:right}#profile #metadata #people h2{font-size:14px;color:#747474}#profile #metadata #people li{display:block;float:left}#profile #metadata #people li+li{margin-left:4px}#profile #metadata #people a{display:block}#profile h2{font-size:16px;line-height:20px;color:#2b547d}#profile #newsfeed{width:430px;padding:30px 35px 0 0}#profile #newsfeed>p{margin-top:6px;font-weight:inherit;line-height:18px}#profile #newsfeed>ol>li{padding-left:0}#profile #newsfeed>ol>li>h4{position:absolute;left:-9999px}#profile #newsfeed>ol>li>h4+a{display:none}#profile #relevant-repos{float:right;width:430px;padding:30px 0 0 35px;font-size:13px}#profile #relevant-repos>ol{margin:10px -10px 0 -12px}#profile #relevant-repos>ol>li{position:relative;padding:10px 10px 10px 12px}#profile #relevant-repos h3{margin-right:100px;font-weight:normal;font-size:18px;line-height:22px;overflow:hidden}#profile #relevant-repos .metadata{margin:0;font-size:11px;line-height:20px;color:#999}#profile #relevant-repos .repository{right:85px}#profile #relevant-repos .followers{right:47px}#profile #relevant-repos p{margin-top:6px}#promo>.two-column{padding-top:10px}#promo>.two-column h3{margin:18px 0 0;line-height:18px}#pullrequest h1{margin-top:20px;line-height:24px;color:#333}#pullrequest .create-pr-description{margin:1em 0}#pullrequest p.metadata{font-size:11px;color:#999}#pullrequest p.metadata>a{text-decoration:underline;color:inherit}#pullrequest p.info{margin-top:10px}#pullrequest #id_description{height:120px}#pullrequest #pr-create{width:auto}#pullrequest #pr-create>div{width:560px}#pullrequest #pull-from,#pullrequest #pull-into{margin:12px 0;min-height:62px;border:1px solid #ddd;background:#f9f9f9;padding:10px 12px 12px 58px}#pullrequest #pull-from.deleted{padding-left:12px;padding-right:58px}#pullrequest div#pull-from{float:left;width:488px}#pullrequest div#pull-into{clear:none;float:right;margin-left:38px;width:260px}#pullrequest #pull-into:before{position:absolute;left:-29px;top:54px;width:19px;height:19px;background:url();content:""}#pullrequest #pull-from h3,#pullrequest #pull-into h3{margin:0;padding:0;font-weight:normal;font-size:16px;color:#999}#pullrequest #pull-from h3 a,#pullrequest #pull-into h3 a{line-height:22px;color:inherit}#pullrequest #pull-from h3 a~a,#pullrequest #pull-into h3 a~a{color:#2b547d}#pullrequest #pull-from img.avatar,#pullrequest #pull-into img.avatar{position:absolute;left:10px;top:10px}#pullrequest #pull-from p.metadata,#pullrequest #pull-into p.metadata{margin-bottom:0;font-size:11px;line-height:14px;color:#999}#pullrequest #pull-from p.metadata+p{margin-top:8px}#pullrequest .bb-dropdown.open li[data-exists="false"]{display:none}#pullrequest .bb-dropdown.open li[data-exists="false"].new-branch{display:block}#pullrequest .bb-dropdown.open li[data-exists="false"].new-branch>h4:after{content:"\0020+ new branch";color:#999}#pullrequest .manual-merge-required{display:none}#pullrequest #pull-from.suppress-errors .error_.field_.message_,#pullrequest #pull-into.suppress-errors .error_.field_.message_{display:none}#pullrequest .newform #pull-from .field_.message_,#pullrequest .newform #pull-into .field_.message_{margin-top:10px;margin-bottom:0}#pullrequest .branch-summary{display:block;margin:3px 0 0;height:39px;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;border:1px solid #ddd;background:#eee;padding:9px 12px;line-height:30px;overflow:hidden}#pullrequest #dest-branch.loading{background:#fff6c2}#pullrequest .branch-dropdown{color:#999}#pullrequest .branch-dropdown>li{padding-top:0}#pullrequest .branch-summary>h4,#pullrequest .branch-dropdown>li>h4{display:inline;margin:0;padding:0;font-weight:normal;font-size:13px;color:#666}#pullrequest .branch-summary>h4>a~a,#pullrequest .branch-dropdown>li>h4>a~a{color:inherit}#pullrequest .branch-summary>h4 code,#pullrequest .branch-dropdown>li>h4 code{border:0;background:0;padding:0 1px}#pullrequest .branch-summary>p,#pullrequest .branch-dropdown>li>p{display:inline}#pullrequest .branch-summary>p>a,#pullrequest .branch-dropdown>li>p>a{color:inherit}#pullrequest .branch-summary>p:before,#pullrequest .branch-dropdown>li>p:before{content:"~\0020"}#pullrequest .branch-summary>p:after,#pullrequest .branch-dropdown>li>p:after{content:":"}#pullrequest .branch-summary>blockquote,#pullrequest .branch-dropdown>li>blockquote{display:inline}#pullrequest #summary{clear:both}#pullrequest #compare{position:relative}#pullrequest #compare #compare-header{display:none}#pullrequest #compare.loading+#mask{position:absolute;top:0;left:0;right:0;bottom:0;background:rgba(255,255,255,0.5)}#pullrequest #compare>hr{clear:both;margin:0;height:0;border:0;border-top:2px solid #ddd}#pullrequest[class] .list>h1{text-transform:lowercase}#pullrequest .list>h1:before{text-transform:none}#pullrequest .view #compare>hr{display:none}#pullrequest #compare .cross-repo-merge{display:none}#pullrequest .view .followers{position:absolute;right:37px;top:19px}#pullrequest .view.can-fulfill .followers{right:181px}#pullrequest .view .followers .info-button-label{font-weight:bold;color:#666}#pullrequest #pr-statuses.button-group{position:absolute;right:0;margin-top:-30px}#pullrequest .list .pr-list{margin-top:10px}#pullrequest .pr-list th,#pullrequest .pr-list td{width:400px}#pullrequest .pr-list th+th,#pullrequest .pr-list td.repo.deleted{width:270px}#pullrequest .pr-list td+td{width:242px}#pullrequest .pr-list th+th+th,#pullrequest .pr-list td+td+td{width:90px}#pullrequest .pr-list th+th+th.open-datetime,#pullrequest .pr-list td+td+td.open-datetime{width:200px}#pullrequest .pr-list td[colspan="3"],#pullrequest .pr-list td[colspan="4"]{width:910px;font-style:italic}#pullrequest.open h1:before{content:"Open\0020"}#pullrequest.accepted h1:before{content:"Accepted\0020"}#pullrequest.rejected h1:before{content:"Rejected\0020"}#pullrequest.open .button-group a[href="#open"],#pullrequest.accepted .button-group a[href="#accepted"],#pullrequest.rejected .button-group a[href="#rejected"],#pullrequest .button-group a:active{background:#ccc;background:-webkit-gradient(linear,0 0,0 100%,from(#aaa),to(#eee),color-stop(0.25,#ccc));background:-webkit-linear-gradient(#aaa,#ccc 25%,#eee);background:-moz-linear-gradient(top,#aaa,#ccc 25%,#eee);background:-o-linear-gradient(#aaa,#ccc 25%,#eee);color:#393939}#pullrequest .view>.metadata{font-size:12px}#pullrequest .view.can-fulfill>h1{margin-bottom:0}#pullrequest .view .conflicts-detected,#pullrequest .view .multiple-destination-heads,#pullrequest .view .accepted,#pullrequest .view .rejected,#pullrequest .view .superseded{margin:15px 0 12px;border-width:1px;border-style:solid;padding:0 24px}#pullrequest .view .conflicts-detected,#pullrequest .view .multiple-destination-heads{border-color:#fc0;border-color:rgba(255,204,0,0.6);background:rgba(255,204,0,0.08)}#pullrequest .view .conflicts-detected{display:none}#pullrequest.conflicts .view .conflicts-detected{display:inherit}#pullrequest.conflicts #pr-fulfill{color:#666;cursor:default}#pullrequest.conflicts #pr-fulfill:active{background:#eee;background:-webkit-gradient(linear,0 0,0 100%,from(#fff),to(#ddd));background:-webkit-linear-gradient(#fff,#ddd);background:-moz-linear-gradient(top,#fff,#ddd);background:-o-linear-gradient(#fff,#ddd)}#pullrequest.conflicts #pr-fulfill:before{background-position:0 -16px}#pullrequest .view .accepted{border-color:#282;border-color:rgba(34,136,34,0.4);background:rgba(34,136,34,0.02)}#pullrequest .view .rejected{border-color:#c00;border-color:rgba(204,0,0,0.3);background:rgba(204,0,0,0.02)}#pullrequest .view .superseded{border-color:#2b547d;border-color:rgba(43,84,125,0.28);background:rgba(234,250,255,0.3)}#pullrequest .view h2.state{display:inline-block;padding:6px 0}#pullrequest .view .conflicts-detected h2.state,#pullrequest .view .multiple-destination-heads h2.state{padding-bottom:0;color:#333}#pullrequest .view .accepted h2.state{color:#282}#pullrequest .view .rejected h2.state{color:#c00}#pullrequest .view h2.state+p.metadata{display:inline;font-size:12px}#pullrequest .view .conflicts-detected>pre{margin-bottom:18px;border-color:#e7e7e7;background:#f7f7f7}#pullrequest .view .resolution{margin:-6px 0 12px}#pullrequest .view label{display:block;padding:16px 0 4px;font-weight:bold;color:#666}#pullrequest .view #summary{margin-top:-13px}#pullrequest #pr-description{clear:both}#pullrequest .view .bb-dropdown li[data-exists="false"]>h4:before{content:"+\0020"}#pullrequest .view #pull-from,#pullrequest .view #pull-into{min-height:51px;margin-bottom:8px;padding-bottom:8px}#pullrequest .view #pull-into:before{top:26px}#pullrequest .view #pull-from>h3,#pullrequest .view #pull-into>h3{float:left;margin-top:-1px;margin-bottom:2px}#pullrequest .view #pull-from.deleted>h3{float:none;margin-top:2px;margin-bottom:12px}#pullrequest .view #pull-from>.metadata{float:left;margin-left:4px;margin-top:5px}#pullrequest .view .bb-dropdown{clear:both}#pullrequest .view #pull-into>.metadata,#pullrequest .view #pull-from>label,#pullrequest .view #pull-into>label{display:none}#pullrequest .view .bb-dropdown{border:1px solid #ddd;background:#fcfcfc}#pullrequest .view .branch-dropdown{margin:2px;border:0;background:0}#pullrequest .view .branch-dropdown>li{margin:0;border:0}#pullrequest .list .pr-list{display:none}#pullrequest.open .pr-list.open,#pullrequest.accepted .pr-list.accepted,#pullrequest.rejected .pr-list.rejected{display:table}#pullrequest #pr-actions{float:right;margin:0 0 10px 20px}#pullrequest #pr-actions .bb-button{width:60px}#pullrequest #pr-description pre{border:1px solid #ddd;background:#eee;padding:0 2px;font-size:12px}#pullrequest #pr-description pre{padding:4px 8px}#pullrequest #pr-description pre>code{border:0;background:0;padding:0}#id_reason{height:100px}#id_commit_message{height:20px}#id_commit_message.expanded{height:100px}#pullrequest #pullrequest-actions .update-tooltip{display:none}#pullrequest .loading h3{background:url(../img/loading.gif) no-repeat 0 11px;padding-left:20px}#pullrequest #comments-count{color:#aaa}#search #search-form_{margin-bottom:20px;border-bottom:1px solid #e7e7e7;padding-bottom:10px;position:relative}#search #search-form_>div{float:left}#search #search-text{display:block;float:left;margin:0;width:650px;height:22px;padding:4px 5px 4px 30px;font-size:18px;outline:0;line-height:20px;border:1px solid #ddd;background:url(../images/search-docs-white-bg-30px.png) no-repeat 0 -1px;background-color:white;-webkit-appearance:none}#search #search-button{display:block;float:left;margin-left:0;margin-right:5px;height:32px;border:1px solid #ddd;padding:0 10px;font-size:15px}#search #search-form_ label{font-weight:normal}#search #search-form_ select{margin:0}#search #search-form_ .operand{min-width:215px}#search #search-form_ #results-count{background:#fff;bottom:-10px;clear:both;color:#999;font-size:12px;font-style:italic;left:44px;padding:0 6px;position:absolute}#search #search-form_ div.clearer{clear:both;float:none;font-size:1px;margin:0}#search #repositories>section{padding:10px 0 10px 50px}#search #repositories>section>.repository{margin:0;font-size:15px;line-height:1.2;color:#999;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;width:72%}#search #repositories>section>.repository>.owner{color:inherit}#search #repositories>section.repository>h1>a,#search #repositories>section.user>h1>a>span{color:#2b547d}#search #repositories>section.repository>h1>.owner{color:inherit}#search #repositories>section>.avatar{position:absolute;left:0;top:10px;display:block;width:32px;height:32px;border:1px solid #e7e7e7;padding:2px}#search #repositories>section>.avatar:after{position:absolute;left:27px;top:27px;display:block;width:16px;height:16px;content:""}#search #repositories .resulttype{color:#999;font-size:15px}#search #repositories>section.commit>.avatar:after{background:url(../img/icons/puzzle.png)}#search #repositories>section.user>.avatar:after{background:url(../img/icons/user.png)}#search #repositories>section.private.repository>.avatar:after{background:url(../img/icons/droplets-16.png)}#search #repositories>section.public.repository>.avatar:after{background:url(../img/icons/droplets-16.png)}#search #repositories>section.source>.avatar:after{background:url(../img/icons/source.png)}#search #repositories>section>.avatar>img{width:100%;height:100%}#search #repositories>section.commit>.message,#search #repositories>section.repository>.message,#search #repositories>section.user>.message{border:0;text-align:left}#search #repositories>section.commit>.message,#search #repositories>section.user>.message,#search #repositories>section.repository>.message,#search #repositories>section.source>pre{clear:both;margin:5px 0 4px;padding:1px}#search #repositories>section.commit>.message>div,#search #repositories>section.user>.message>div,#search #repositories>section.repository>.message>div,#search #repositories>section.source>pre{clear:both;background:#fff;padding:4px 6px;white-space:pre-wrap;line-height:16px;overflow:hidden;text-overflow:ellipsis}#search #repositories>section.commit>.message>div>h4{margin:0;font-weight:normal}#search #repositories>section .metadata{float:none;margin:0;font-size:11px;line-height:20px;color:#999}#search #repositories>section em{font-style:normal;font-weight:bold;color:#000}#search .error{color:#999}#search .searchInput{margin-bottom:10px}#search .searchInput .selectors-wrap{clear:both;float:left;margin-right:.5em}#search .actions-wrap{float:right}#search .actions-wrap input{margin-left:.5em;margin-top:-0.2em}#search #advanced_link{float:right;font-size:12px;padding:2px 4px}#search #sort-repos{background:#fff;bottom:-10px;display:none;padding-left:10px;position:absolute;right:0}#search .errorContainer{clear:left}#search #advanced-input{clear:both;margin-top:8px;width:510px}#search #mask.loading{position:absolute;top:0;left:0;right:0;bottom:0;background:rgba(255,255,255,0.5)}#search .location{position:absolute;left:5px;top:5px;background:url(../img/icons/location-pin.png) no-repeat;padding-left:20px}#search dd.branches .info-button .info-button-action a{background-image:url(../img/icons/fugue/arrow_135.png)}#search dd.tags .info-button .info-button-action a{background-image:url(../img/icons/fugue/tags.png)}#search dd.tags,#search dd.user-followers,#search dd.branches{position:absolute;top:10px}#search dd.branches{right:86px}#search dd.user-followers{right:47px}#search dd.tags{right:205px}#search dd.user-followers .info-button-readonly .info-button-action a,#search dd.user-followers-not-following .info-button .info-button-action a{background-image:url(../img/icons/fugue/heart_empty.png)}#search dd.user-followers-not-following .info-button .info-button-action a:hover,#search dd.user-followers-following .info-button .info-button-action a{background-image:url(../img/icons/fugue/heart.png)}#search dd.user-followers-following .info-button .info-button-action a:hover{background-image:url(../img/icons/fugue/heart_break.png)}#search dd.user-followers .info-button-requesting .info-button-action a,#search dd.user-followers .info-button-requesting .info-button-action a:hover{background-image:url(../img/loading.gif)}#search .tags{float:none}#search .def-only dd ul{width:75px}#search .def-only dd .info-button-label{background:#fff}#search .def-only dd:hover .info-button-label{z-index:1;border-radius:0 5px 0 0;-moz-border-radius:0 5px 0 0;-webkit-border-radius:0 5px 0 0}#search .def-only dd li+li{display:none;border-top:1px solid #ddd;text-align:left}#search .def-only dd:hover li+li{display:block}#search .def-only dd li{display:block;padding:0 2px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}#search .def-only dd li+li{border-top:1px solid #ddd}#search .def-only dd li a{margin:0 2px;width:auto}#search .def-only>ul>li>a{display:block;text-decoration:none}#search dd.tags .info-button-label,dd.branches .info-button-label{width:auto}#search dd.privacy .private,#search dd.privacy .public{position:absolute;right:251px;top:14px;width:16px;height:16px;background:url(../img/icons/fugue/lock.png) no-repeat;text-indent:-9999px}#search dd.privacy .public{top:13px;background-image:url(../img/icons/fugue/lock_disable_unlock.png)}#search .button-group{float:left;margin-right:10px}#search .search-type{position:relative;top:2px}#search .second-row-input{clear:both;width:760px}#search .search-type-container{width:130px}#search .bb-dropdown>ul{margin:-4px}#signup #openid_form input[type="text"],#signup #openid_form input[type="url"]{width:286px}#signup .two-thirds-secondary{margin-top:31px}#signup .two-thirds-secondary>h3{margin-bottom:12px}#signup .two-thirds-secondary>ul{line-height:20px}#signup .two-thirds-secondary>ul>li{position:relative;padding-left:20px}#signup .two-thirds-secondary>ul>li:before{position:absolute;top:8px;margin:0 8px 0 -12px;vertical-align:middle;display:inline-block;width:4px;height:4px;background:#999;content:""}#source>#sourcelist{clear:both;margin-bottom:25px}#source>h3{margin-left:25px}#source .message_.note_{margin:10px 0;background-image:url(../img/icons/info.png);background-repeat:no-repeat;background-position:11px}#strip-repository .newform .submit>a{margin-left:5px;font-size:12px}#strip-repository .newtable .hash{width:100px}#strip-repository .newtable .text{width:490px}#wiki{border:1px solid #ddd;padding:15px 24px;font-size:14px;line-height:1.5em;color:#444}#wiki .wiki-edit-table th{vertical-align:middle}.wiki-edit-table #id_message,#wiki .wiki-edit-table #id_message,#wiki .wiki-edit-table #id_path{margin:inherit;width:690px;height:inherit;padding:3px 5px}#wiki-nav-links{margin:0 0 10px 25px;line-height:21px;overflow:auto}#wiki-nav-links li{float:left;margin:0 15px 0 0}#wiki-nav-links li.home,#wiki-nav-links li.page{font-weight:bold}#wiki-nav-links li.new{display:none}#wiki-nav-links a{background-image:url(../img/icons/fugue/document_small.png);background-repeat:no-repeat;background-position:0 50%;padding:0 0 0 17px}#wiki-nav-links li.home a{background-image:url(../img/icons/fugue/home_small.png)}#wiki-nav-links li.edit a{background-image:url(../img/icons/fugue/pencil_small.png)}#wiki-nav-links li.history a{background-image:url(../img/icons/fugue/clock_small.png)}#wiki-nav-links li.new a{background-image:url(../img/icons/fugue/plus_small.png)}p.clone-wiki{float:right;margin:0 25px 0 0;line-height:21px}#wiki-new-page-form{display:none;margin:0 0 10px 25px}#wiki h1{margin:.67em 0;font-size:2.5em}#wiki h2{margin:.83em 0;font-size:2em}#wiki h3{margin:1em 0;font-size:1.5em}#wiki h4,#wiki p,#wiki blockquote,#wiki ul,#wiki fieldset,#wiki form,#wiki ol,#wiki dl,#wiki dir,#wiki menu{margin:1.33em 0}#wiki ul li>ul{margin-left:13px}#wiki ol li>ol{margin-left:18px}#wiki p{padding:0}#wiki h5{margin:1.67em 0;font-size:.83em;line-height:1.17em}#wiki h6{margin:2.33em 0;font-size:.67em}#wiki h1,#wiki h2,#wiki h3,#wiki h4,#wiki h5,#wiki h6{font-weight:normal}#wiki table.wikitable{border-collapse:collapse}#wiki table.wikitable td{border-left:1px solid #b7b7b7;border-right:1px solid #b7b7b7;border-bottom:1px solid #b7b7b7;padding:6px 12px;font:11px "Lucida Grande",Verdana,Arial,sans-serif}#wiki table.wikitable th{border:1px solid #b7b7b7;background:#fff url(../img/wiki/bg-th.png) no-repeat left top;padding:6px 12px;text-align:left;font:bold 11px "Lucida Grande",Verdana,Arial,sans-serif;text-transform:uppercase;letter-spacing:1px;color:#2b547d}#wiki blockquote{margin-left:40px;margin-right:40px}#wiki .floatleft{float:left;width:50%}#wiki .floatright{float:right;width:50%}#wiki i,#wiki cite,#wiki em,#wiki var,#wiki address{font-style:italic}#wiki tt,#wiki kbd,#wiki samp{font-family:monospace}#wiki pre{margin:9px 0;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;border:1px solid #ddd;background:#f7f7f7;padding:9px 12px;font-size:12px;line-height:16px;white-space:pre-wrap}#wiki big{font-size:1.17em}#wiki small,#wiki sub,#wiki sup{font-size:.83em}#wiki sub{vertical-align:sub}#wiki sup{vertical-align:super}#wiki s,#wiki strike,#wiki del{text-decoration:line-through}#wiki hr{border:1px inset}#wiki dir,#wiki menu,#wiki dd{margin-left:40px}#wiki ul,.issues-issue-description ul{list-style:square inside url()}#wiki ol{position:relative;counter-reset:ol;list-style:none}#wiki ol>li{counter-increment:ol}#wiki ol>li:before{position:absolute;left:-24px;display:block;width:20px;padding:0 4px 0 0;text-align:right;content:counter(ol) "."}#wiki ol ul,#wiki ul ol,#wiki ul ul,#wiki ol ol{margin:0 0 0 24px}#wiki u,#wiki ins{text-decoration:underline}#wiki center{text-align:center}#wiki br:before{content:"\a"}#wiki>.highlighttable{margin:0 -24px -15px;width:auto}.revision-date{text-align:right;font-size:12px}#wiki-history .paginator{margin-top:9px}#wiki-history h1{margin:0;font-weight:normal;font-size:18px;line-height:2;color:#666}#wiki-history .newtable{clear:both}#wiki-history .newtable .hash{width:100px}#wiki-history .newtable .text{width:490px}#wiki-history .newtable .text a{color:inherit}#zealots h1{margin-top:20px}#zealots #zealots-list{margin:10px 0 0;padding:0 0 0 42px;color:#999;list-style:none}#zealots #zealots-list>li{position:relative;margin:4px 0;border:1px solid #eee;padding:7px 0 7px 10px}#zealots .avatar{position:absolute;left:-43px;top:-1px}#zealots .name{font-size:18px;line-height:22px}#zealots .full.name>a~a{color:inherit}#zealots .location{position:absolute;right:20px;top:10px;background:url() no-repeat;padding-left:20px}.ui-helper-hidden{display:none}.ui-helper-hidden-accessible{position:absolute!important;clip:rect(1px 1px 1px 1px);clip:rect(1px,1px,1px,1px)}.ui-helper-reset{margin:0;padding:0;border:0;outline:0;line-height:1.3;text-decoration:none;font-size:100%;list-style:none}.ui-helper-clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}.ui-helper-clearfix{display:inline-block}/*\*/* html .ui-helper-clearfix{height:1%}.ui-helper-clearfix{display:block}/**/.ui-helper-zfix{width:100%;height:100%;top:0;left:0;position:absolute;opacity:0;filter:Alpha(Opacity=0)}.ui-state-disabled{cursor:default!important}.ui-icon{display:block;text-indent:-99999px;overflow:hidden;background-repeat:no-repeat}.ui-widget-overlay{position:absolute;top:0;left:0;width:100%;height:100%}.ui-widget{font-family:Lucida Grande,Lucida Sans,Arial,sans-serif;font-size:1.1em}.ui-widget .ui-widget{font-size:1em}.ui-widget input,.ui-widget select,.ui-widget textarea,.ui-widget button{font-family:Lucida Grande,Lucida Sans,Arial,sans-serif;font-size:1em}.ui-widget-content{border:1px solid #ddd;background:#f2f5f7 url(../../images/jqueryui/ui-bg_highlight-hard_100_f2f5f7_1x100.png) 50% top repeat-x;color:#362b36}.ui-widget-content a{color:#362b36}.ui-widget-header{border:1px solid #aed0ea;background:#deedf7 url(../../images/jqueryui/ui-bg_highlight-soft_100_deedf7_1x100.png) 50% 50% repeat-x;color:#222;font-weight:bold}.ui-widget-header a{color:#222}.ui-state-default,.ui-widget-content .ui-state-default,.ui-widget-header .ui-state-default{border:1px solid #aed0ea;background:#d7ebf9 url(../../images/jqueryui/ui-bg_glass_80_d7ebf9_1x400.png) 50% 50% repeat-x;font-weight:bold;color:#2779aa}.ui-state-default a,.ui-state-default a:link,.ui-state-default a:visited{color:#2779aa;text-decoration:none}.ui-state-hover,.ui-widget-content .ui-state-hover,.ui-widget-header .ui-state-hover,.ui-state-focus,.ui-widget-content .ui-state-focus,.ui-widget-header .ui-state-focus{border:1px solid #74b2e2;background:#e4f1fb url(../../images/jqueryui/ui-bg_glass_100_e4f1fb_1x400.png) 50% 50% repeat-x;font-weight:bold;color:#0070a3}.ui-state-hover a,.ui-state-hover a:hover{color:#0070a3;text-decoration:none}.ui-state-active,.ui-widget-content .ui-state-active,.ui-widget-header .ui-state-active{border:1px solid #2694e8;background:#3baae3 url(../../images/jqueryui/ui-bg_glass_50_3baae3_1x400.png) 50% 50% repeat-x;font-weight:bold;color:#fff}.ui-state-active a,.ui-state-active a:link,.ui-state-active a:visited{color:#fff;text-decoration:none}.ui-widget :active{outline:0}.ui-state-highlight,.ui-widget-content .ui-state-highlight,.ui-widget-header .ui-state-highlight{border:1px solid #f9dd34;background:#ffef8f url(../../images/jqueryui/ui-bg_highlight-soft_25_ffef8f_1x100.png) 50% top repeat-x;color:#363636}.ui-state-highlight a,.ui-widget-content .ui-state-highlight a,.ui-widget-header .ui-state-highlight a{color:#363636}.ui-state-error,.ui-widget-content .ui-state-error,.ui-widget-header .ui-state-error{border:1px solid #cd0a0a;background:#cd0a0a url(../../images/jqueryui/ui-bg_flat_15_cd0a0a_40x100.png) 50% 50% repeat-x;color:#fff}.ui-state-error a,.ui-widget-content .ui-state-error a,.ui-widget-header .ui-state-error a{color:#fff}.ui-state-error-text,.ui-widget-content .ui-state-error-text,.ui-widget-header .ui-state-error-text{color:#fff}.ui-priority-primary,.ui-widget-content .ui-priority-primary,.ui-widget-header .ui-priority-primary{font-weight:bold}.ui-priority-secondary,.ui-widget-content .ui-priority-secondary,.ui-widget-header .ui-priority-secondary{opacity:.7;filter:Alpha(Opacity=70);font-weight:normal}.ui-state-disabled,.ui-widget-content .ui-state-disabled,.ui-widget-header .ui-state-disabled{opacity:.35;filter:Alpha(Opacity=35);background-image:none}.ui-icon{width:16px;height:16px;background-image:url(../../images/jqueryui/ui-icons_72a7cf_256x240.png)}.ui-widget-content .ui-icon{background-image:url(../../images/jqueryui/ui-icons_72a7cf_256x240.png)}.ui-widget-header .ui-icon{background-image:url(../../images/jqueryui/ui-icons_72a7cf_256x240.png)}.ui-state-default .ui-icon{background-image:url(../../images/jqueryui/ui-icons_3d80b3_256x240.png)}.ui-state-hover .ui-icon,.ui-state-focus .ui-icon{background-image:url(../../images/jqueryui/ui-icons_2694e8_256x240.png)}.ui-state-active .ui-icon{background-image:url(../../images/jqueryui/ui-icons_ffffff_256x240.png)}.ui-state-highlight .ui-icon{background-image:url(../../images/jqueryui/ui-icons_2e83ff_256x240.png)}.ui-state-error .ui-icon,.ui-state-error-text .ui-icon{background-image:url(../../images/jqueryui/ui-icons_ffffff_256x240.png)}.ui-icon-carat-1-n{background-position:0 0}.ui-icon-carat-1-ne{background-position:-16px 0}.ui-icon-carat-1-e{background-position:-32px 0}.ui-icon-carat-1-se{background-position:-48px 0}.ui-icon-carat-1-s{background-position:-64px 0}.ui-icon-carat-1-sw{background-position:-80px 0}.ui-icon-carat-1-w{background-position:-96px 0}.ui-icon-carat-1-nw{background-position:-112px 0}.ui-icon-carat-2-n-s{background-position:-128px 0}.ui-icon-carat-2-e-w{background-position:-144px 0}.ui-icon-triangle-1-n{background-position:0 -16px}.ui-icon-triangle-1-ne{background-position:-16px -16px}.ui-icon-triangle-1-e{background-position:-32px -16px}.ui-icon-triangle-1-se{background-position:-48px -16px}.ui-icon-triangle-1-s{background-position:-64px -16px}.ui-icon-triangle-1-sw{background-position:-80px -16px}.ui-icon-triangle-1-w{background-position:-96px -16px}.ui-icon-triangle-1-nw{background-position:-112px -16px}.ui-icon-triangle-2-n-s{background-position:-128px -16px}.ui-icon-triangle-2-e-w{background-position:-144px -16px}.ui-icon-arrow-1-n{background-position:0 -32px}.ui-icon-arrow-1-ne{background-position:-16px -32px}.ui-icon-arrow-1-e{background-position:-32px -32px}.ui-icon-arrow-1-se{background-position:-48px -32px}.ui-icon-arrow-1-s{background-position:-64px -32px}.ui-icon-arrow-1-sw{background-position:-80px -32px}.ui-icon-arrow-1-w{background-position:-96px -32px}.ui-icon-arrow-1-nw{background-position:-112px -32px}.ui-icon-arrow-2-n-s{background-position:-128px -32px}.ui-icon-arrow-2-ne-sw{background-position:-144px -32px}.ui-icon-arrow-2-e-w{background-position:-160px -32px}.ui-icon-arrow-2-se-nw{background-position:-176px -32px}.ui-icon-arrowstop-1-n{background-position:-192px -32px}.ui-icon-arrowstop-1-e{background-position:-208px -32px}.ui-icon-arrowstop-1-s{background-position:-224px -32px}.ui-icon-arrowstop-1-w{background-position:-240px -32px}.ui-icon-arrowthick-1-n{background-position:0 -48px}.ui-icon-arrowthick-1-ne{background-position:-16px -48px}.ui-icon-arrowthick-1-e{background-position:-32px -48px}.ui-icon-arrowthick-1-se{background-position:-48px -48px}.ui-icon-arrowthick-1-s{background-position:-64px -48px}.ui-icon-arrowthick-1-sw{background-position:-80px -48px}.ui-icon-arrowthick-1-w{background-position:-96px -48px}.ui-icon-arrowthick-1-nw{background-position:-112px -48px}.ui-icon-arrowthick-2-n-s{background-position:-128px -48px}.ui-icon-arrowthick-2-ne-sw{background-position:-144px -48px}.ui-icon-arrowthick-2-e-w{background-position:-160px -48px}.ui-icon-arrowthick-2-se-nw{background-position:-176px -48px}.ui-icon-arrowthickstop-1-n{background-position:-192px -48px}.ui-icon-arrowthickstop-1-e{background-position:-208px -48px}.ui-icon-arrowthickstop-1-s{background-position:-224px -48px}.ui-icon-arrowthickstop-1-w{background-position:-240px -48px}.ui-icon-arrowreturnthick-1-w{background-position:0 -64px}.ui-icon-arrowreturnthick-1-n{background-position:-16px -64px}.ui-icon-arrowreturnthick-1-e{background-position:-32px -64px}.ui-icon-arrowreturnthick-1-s{background-position:-48px -64px}.ui-icon-arrowreturn-1-w{background-position:-64px -64px}.ui-icon-arrowreturn-1-n{background-position:-80px -64px}.ui-icon-arrowreturn-1-e{background-position:-96px -64px}.ui-icon-arrowreturn-1-s{background-position:-112px -64px}.ui-icon-arrowrefresh-1-w{background-position:-128px -64px}.ui-icon-arrowrefresh-1-n{background-position:-144px -64px}.ui-icon-arrowrefresh-1-e{background-position:-160px -64px}.ui-icon-arrowrefresh-1-s{background-position:-176px -64px}.ui-icon-arrow-4{background-position:0 -80px}.ui-icon-arrow-4-diag{background-position:-16px -80px}.ui-icon-extlink{background-position:-32px -80px}.ui-icon-newwin{background-position:-48px -80px}.ui-icon-refresh{background-position:-64px -80px}.ui-icon-shuffle{background-position:-80px -80px}.ui-icon-transfer-e-w{background-position:-96px -80px}.ui-icon-transferthick-e-w{background-position:-112px -80px}.ui-icon-folder-collapsed{background-position:0 -96px}.ui-icon-folder-open{background-position:-16px -96px}.ui-icon-document{background-position:-32px -96px}.ui-icon-document-b{background-position:-48px -96px}.ui-icon-note{background-position:-64px -96px}.ui-icon-mail-closed{background-position:-80px -96px}.ui-icon-mail-open{background-position:-96px -96px}.ui-icon-suitcase{background-position:-112px -96px}.ui-icon-comment{background-position:-128px -96px}.ui-icon-person{background-position:-144px -96px}.ui-icon-print{background-position:-160px -96px}.ui-icon-trash{background-position:-176px -96px}.ui-icon-locked{background-position:-192px -96px}.ui-icon-unlocked{background-position:-208px -96px}.ui-icon-bookmark{background-position:-224px -96px}.ui-icon-tag{background-position:-240px -96px}.ui-icon-home{background-position:0 -112px}.ui-icon-flag{background-position:-16px -112px}.ui-icon-calendar{background-position:-32px -112px}.ui-icon-cart{background-position:-48px -112px}.ui-icon-pencil{background-position:-64px -112px}.ui-icon-clock{background-position:-80px -112px}.ui-icon-disk{background-position:-96px -112px}.ui-icon-calculator{background-position:-112px -112px}.ui-icon-zoomin{background-position:-128px -112px}.ui-icon-zoomout{background-position:-144px -112px}.ui-icon-search{background-position:-160px -112px}.ui-icon-wrench{background-position:-176px -112px}.ui-icon-gear{background-position:-192px -112px}.ui-icon-heart{background-position:-208px -112px}.ui-icon-star{background-position:-224px -112px}.ui-icon-link{background-position:-240px -112px}.ui-icon-cancel{background-position:0 -128px}.ui-icon-plus{background-position:-16px -128px}.ui-icon-plusthick{background-position:-32px -128px}.ui-icon-minus{background-position:-48px -128px}.ui-icon-minusthick{background-position:-64px -128px}.ui-icon-close{background-position:-80px -128px}.ui-icon-closethick{background-position:-96px -128px}.ui-icon-key{background-position:-112px -128px}.ui-icon-lightbulb{background-position:-128px -128px}.ui-icon-scissors{background-position:-144px -128px}.ui-icon-clipboard{background-position:-160px -128px}.ui-icon-copy{background-position:-176px -128px}.ui-icon-contact{background-position:-192px -128px}.ui-icon-image{background-position:-208px -128px}.ui-icon-video{background-position:-224px -128px}.ui-icon-script{background-position:-240px -128px}.ui-icon-alert{background-position:0 -144px}.ui-icon-info{background-position:-16px -144px}.ui-icon-notice{background-position:-32px -144px}.ui-icon-help{background-position:-48px -144px}.ui-icon-check{background-position:-64px -144px}.ui-icon-bullet{background-position:-80px -144px}.ui-icon-radio-off{background-position:-96px -144px}.ui-icon-radio-on{background-position:-112px -144px}.ui-icon-pin-w{background-position:-128px -144px}.ui-icon-pin-s{background-position:-144px -144px}.ui-icon-play{background-position:0 -160px}.ui-icon-pause{background-position:-16px -160px}.ui-icon-seek-next{background-position:-32px -160px}.ui-icon-seek-prev{background-position:-48px -160px}.ui-icon-seek-end{background-position:-64px -160px}.ui-icon-seek-start{background-position:-80px -160px}.ui-icon-seek-first{background-position:-80px -160px}.ui-icon-stop{background-position:-96px -160px}.ui-icon-eject{background-position:-112px -160px}.ui-icon-volume-off{background-position:-128px -160px}.ui-icon-volume-on{background-position:-144px -160px}.ui-icon-power{background-position:0 -176px}.ui-icon-signal-diag{background-position:-16px -176px}.ui-icon-signal{background-position:-32px -176px}.ui-icon-battery-0{background-position:-48px -176px}.ui-icon-battery-1{background-position:-64px -176px}.ui-icon-battery-2{background-position:-80px -176px}.ui-icon-battery-3{background-position:-96px -176px}.ui-icon-circle-plus{background-position:0 -192px}.ui-icon-circle-minus{background-position:-16px -192px}.ui-icon-circle-close{background-position:-32px -192px}.ui-icon-circle-triangle-e{background-position:-48px -192px}.ui-icon-circle-triangle-s{background-position:-64px -192px}.ui-icon-circle-triangle-w{background-position:-80px -192px}.ui-icon-circle-triangle-n{background-position:-96px -192px}.ui-icon-circle-arrow-e{background-position:-112px -192px}.ui-icon-circle-arrow-s{background-position:-128px -192px}.ui-icon-circle-arrow-w{background-position:-144px -192px}.ui-icon-circle-arrow-n{background-position:-160px -192px}.ui-icon-circle-zoomin{background-position:-176px -192px}.ui-icon-circle-zoomout{background-position:-192px -192px}.ui-icon-circle-check{background-position:-208px -192px}.ui-icon-circlesmall-plus{background-position:0 -208px}.ui-icon-circlesmall-minus{background-position:-16px -208px}.ui-icon-circlesmall-close{background-position:-32px -208px}.ui-icon-squaresmall-plus{background-position:-48px -208px}.ui-icon-squaresmall-minus{background-position:-64px -208px}.ui-icon-squaresmall-close{background-position:-80px -208px}.ui-icon-grip-dotted-vertical{background-position:0 -224px}.ui-icon-grip-dotted-horizontal{background-position:-16px -224px}.ui-icon-grip-solid-vertical{background-position:-32px -224px}.ui-icon-grip-solid-horizontal{background-position:-48px -224px}.ui-icon-gripsmall-diagonal-se{background-position:-64px -224px}.ui-icon-grip-diagonal-se{background-position:-80px -224px}.ui-corner-tl{-moz-border-radius-topleft:6px;-webkit-border-top-left-radius:6px;border-top-left-radius:6px}.ui-corner-tr{-moz-border-radius-topright:6px;-webkit-border-top-right-radius:6px;border-top-right-radius:6px}.ui-corner-bl{-moz-border-radius-bottomleft:6px;-webkit-border-bottom-left-radius:6px;border-bottom-left-radius:6px}.ui-corner-br{-moz-border-radius-bottomright:6px;-webkit-border-bottom-right-radius:6px;border-bottom-right-radius:6px}.ui-corner-top{-moz-border-radius-topleft:6px;-webkit-border-top-left-radius:6px;border-top-left-radius:6px;-moz-border-radius-topright:6px;-webkit-border-top-right-radius:6px;border-top-right-radius:6px}.ui-corner-bottom{-moz-border-radius-bottomleft:6px;-webkit-border-bottom-left-radius:6px;border-bottom-left-radius:6px;-moz-border-radius-bottomright:6px;-webkit-border-bottom-right-radius:6px;border-bottom-right-radius:6px}.ui-corner-right{-moz-border-radius-topright:6px;-webkit-border-top-right-radius:6px;border-top-right-radius:6px;-moz-border-radius-bottomright:6px;-webkit-border-bottom-right-radius:6px;border-bottom-right-radius:6px}.ui-corner-left{-moz-border-radius-topleft:6px;-webkit-border-top-left-radius:6px;border-top-left-radius:6px;-moz-border-radius-bottomleft:6px;-webkit-border-bottom-left-radius:6px;border-bottom-left-radius:6px}.ui-corner-all{-moz-border-radius:6px;-webkit-border-radius:6px;border-radius:6px}.ui-widget-overlay{background:#eee url(../../images/jqueryui/ui-bg_diagonals-thick_90_eeeeee_40x40.png) 50% 50% repeat;opacity:.80;filter:Alpha(Opacity=80)}.ui-widget-shadow{margin:-7px 0 0 -7px;padding:7px;background:#000 url(../../images/jqueryui/ui-bg_highlight-hard_70_000000_1x100.png) 50% top repeat-x;opacity:.30;filter:Alpha(Opacity=30);-moz-border-radius:8px;-webkit-border-radius:8px;border-radius:8px}.ui-autocomplete{position:absolute;cursor:default}* html .ui-autocomplete{width:1px}.ui-menu{list-style:none;padding:2px;margin:0;display:block;float:left}.ui-menu .ui-menu{margin-top:-3px}.ui-menu .ui-menu-item{margin:0;padding:0;zoom:1;float:left;clear:left;width:100%}.ui-menu .ui-menu-item a{text-decoration:none;display:block;padding:.2em .4em;line-height:1.5;zoom:1}.ui-menu .ui-menu-item a.ui-state-hover,.ui-menu .ui-menu-item a.ui-state-active{font-weight:normal;margin:-1px}.ui-dialog{position:absolute;padding:.2em;width:300px;overflow:hidden}.ui-dialog .ui-dialog-titlebar{padding:.5em 1em .3em;position:relative}.ui-dialog .ui-dialog-title{float:left;margin:.1em 16px .2em 0}.ui-dialog .ui-dialog-titlebar-close{position:absolute;right:.3em;top:50%;width:19px;margin:-10px 0 0 0;padding:1px;height:18px}.ui-dialog .ui-dialog-titlebar-close span{display:block;margin:1px}.ui-dialog .ui-dialog-titlebar-close:hover,.ui-dialog .ui-dialog-titlebar-close:focus{padding:0}.ui-dialog .ui-dialog-content{position:relative;border:0;padding:.5em 1em;background:0;overflow:auto;zoom:1}.ui-dialog .ui-dialog-buttonpane{text-align:left;border-width:1px 0 0 0;background-image:none;margin:.5em 0 0 0;padding:.3em 1em .5em .4em}.ui-dialog .ui-dialog-buttonpane .ui-dialog-buttonset{float:right}.ui-dialog .ui-dialog-buttonpane button{margin:.5em .4em .5em 0;cursor:pointer}.ui-dialog .ui-resizable-se{width:14px;height:14px;right:3px;bottom:3px}.ui-draggable .ui-dialog-titlebar{cursor:move}.ui-widget{font-family:inherit;font-size:inherit}.ui-widget .ui-widget{font-size:inherit}.ui-widget input,.ui-widget select,.ui-widget textarea,.ui-widget button{font-family:inherit;font-size:inherit}.ui-widget-header{border:0;background:#fff url(../../img/wiki/bg-th-blue.png) no-repeat -1px -1px;color:#133362;font-size:14px;font-weight:normal}.ui-widget-overlay{background:#eee}.ui-widget-header.ui-corner-all{-moz-border-radius:6px 6px 0 0;-webkit-border-radius:6px 6px 0 0;border-radius:6px 6px 0 0}.ui-autocomplete.ui-widget-content{background:#fff 50% top repeat-x;color:#333}.ui-autocomplete.ui-widget-content a{color:#333}.ui-autocomplete.ui-widget-header{border:1px solid #f5f5f5;background:#f5f5f5 50% 50% repeat-x;color:#000}.ui-autocomplete.ui-widget-header a{color:#fff}.ui-autocomplete .ui-state-default,.ui-widget-content .ui-state-default,.ui-widget-header .ui-state-default{border:1px solid #ccc;background:#eee 50% 50% repeat-x;color:#1c94c4}.ui-autocomplete .ui-state-default a,.ui-state-default a:link,.ui-state-default a:visited{color:#1c94c4}.ui-autocomplete .ui-state-hover,.ui-widget-content .ui-state-hover,.ui-widget-header .ui-state-hover,.ui-state-focus,.ui-widget-content .ui-state-focus,.ui-widget-header .ui-state-focus{border:1px solid #ddd;background:#eee 50% 50% repeat-x;color:#000}.ui-autocomplete .ui-state-hover a,.ui-state-hover a:hover{color:#f5f5f5}.ui-autocomplete .ui-state-active,.ui-widget-content .ui-state-active,.ui-widget-header .ui-state-active{border:1px solid #eee;background:#fff 50% 50% repeat-x;color:#000}.ui-autocomplete .ui-state-active a,.ui-state-active a:link,.ui-state-active a:visited{color:#000}.ui-autocomplete{max-width:400px;overflow-x:hidden;white-space:nowrap}.ui-menu{padding:1px 0 1px 0}.ui-dialog{padding:inherit}.ui-dialog .ui-dialog-buttonpane{border-width:0;margin:inherit}.ui-datepicker{width:17em;padding:.2em .2em 0;display:none}.ui-datepicker .ui-datepicker-header{position:relative;padding:.2em 0}.ui-datepicker .ui-datepicker-prev,.ui-datepicker .ui-datepicker-next{position:absolute;top:2px;width:1.8em;height:1.8em}.ui-datepicker .ui-datepicker-prev-hover,.ui-datepicker .ui-datepicker-next-hover{top:1px}.ui-datepicker .ui-datepicker-prev{left:2px}.ui-datepicker .ui-datepicker-next{right:2px}.ui-datepicker .ui-datepicker-prev-hover{left:1px}.ui-datepicker .ui-datepicker-next-hover{right:1px}.ui-datepicker .ui-datepicker-prev span,.ui-datepicker .ui-datepicker-next span{display:block;position:absolute;left:50%;margin-left:-8px;top:50%;margin-top:-8px}.ui-datepicker .ui-datepicker-title{margin:0 2.3em;line-height:1.8em;text-align:center}.ui-datepicker .ui-datepicker-title select{font-size:1em;margin:1px 0}.ui-datepicker select.ui-datepicker-month-year{width:100%}.ui-datepicker select.ui-datepicker-month,.ui-datepicker select.ui-datepicker-year{width:49%}.ui-datepicker table{width:100%;font-size:.9em;border-collapse:collapse;margin:0 0 .4em}.ui-datepicker th{padding:.7em .3em;text-align:center;font-weight:bold;border:0}.ui-datepicker td{border:0;padding:1px}.ui-datepicker td span,.ui-datepicker td a{display:block;padding:.2em;text-align:right;text-decoration:none}.ui-datepicker .ui-datepicker-buttonpane{background-image:none;margin:.7em 0 0 0;padding:0 .2em;border-left:0;border-right:0;border-bottom:0}.ui-datepicker .ui-datepicker-buttonpane button{float:right;margin:.5em .2em .4em;cursor:pointer;padding:.2em .6em .3em .6em;width:auto;overflow:visible}.ui-datepicker .ui-datepicker-buttonpane button.ui-datepicker-current{float:left}.ui-datepicker.ui-datepicker-multi{width:auto}.ui-datepicker-multi .ui-datepicker-group{float:left}.ui-datepicker-multi .ui-datepicker-group table{width:95%;margin:0 auto .4em}.ui-datepicker-multi-2 .ui-datepicker-group{width:50%}.ui-datepicker-multi-3 .ui-datepicker-group{width:33.3%}.ui-datepicker-multi-4 .ui-datepicker-group{width:25%}.ui-datepicker-multi .ui-datepicker-group-last .ui-datepicker-header{border-left-width:0}.ui-datepicker-multi .ui-datepicker-group-middle .ui-datepicker-header{border-left-width:0}.ui-datepicker-multi .ui-datepicker-buttonpane{clear:left}.ui-datepicker-row-break{clear:both;width:100%}.ui-datepicker-rtl{direction:rtl}.ui-datepicker-rtl .ui-datepicker-prev{right:2px;left:auto}.ui-datepicker-rtl .ui-datepicker-next{left:2px;right:auto}.ui-datepicker-rtl .ui-datepicker-prev:hover{right:1px;left:auto}.ui-datepicker-rtl .ui-datepicker-next:hover{left:1px;right:auto}.ui-datepicker-rtl .ui-datepicker-buttonpane{clear:right}.ui-datepicker-rtl .ui-datepicker-buttonpane button{float:left}.ui-datepicker-rtl .ui-datepicker-buttonpane button.ui-datepicker-current{float:right}.ui-datepicker-rtl .ui-datepicker-group{float:right}.ui-datepicker-rtl .ui-datepicker-group-last .ui-datepicker-header{border-right-width:0;border-left-width:1px}.ui-datepicker-rtl .ui-datepicker-group-middle .ui-datepicker-header{border-right-width:0;border-left-width:1px}.ui-datepicker-cover{display:none;display:block;position:absolute;z-index:-1;filter:mask();top:-4px;left:-4px;width:200px;height:200px}.ui-progressbar{height:2em;text-align:left}.ui-progressbar .ui-progressbar-value{margin:-1px;height:100%}#fancybox-loading{position:fixed;top:50%;left:50%;width:40px;height:40px;margin-top:-20px;margin-left:-20px;cursor:pointer;overflow:hidden;z-index:1104;display:none}#fancybox-loading div{position:absolute;top:0;left:0;width:40px;height:480px;background-image:url('../../../img/lib/fancybox/fancybox.png')}#fancybox-overlay{position:absolute;top:0;left:0;width:100%;z-index:1100;display:none}#fancybox-tmp{padding:0;margin:0;border:0;overflow:auto;display:none}#fancybox-wrap{position:absolute;top:0;left:0;padding:20px;z-index:1101;outline:0;display:none}#fancybox-outer{position:relative;width:100%;height:100%;background:#fff}#fancybox-content{width:0;height:0;padding:0;outline:0;position:relative;overflow:hidden;z-index:1102;border:0 solid #fff}#fancybox-hide-sel-frame{position:absolute;top:0;left:0;width:100%;height:100%;background:transparent;z-index:1101}#fancybox-close{position:absolute;top:-15px;right:-15px;width:30px;height:30px;background:transparent url('../../../img/lib/fancybox/fancybox.png') -40px 0;cursor:pointer;z-index:1103;display:none}#fancybox-error{color:#444;font:normal 12px/20px Arial;padding:14px;margin:0}#fancybox-img{width:100%;height:100%;padding:0;margin:0;border:0;outline:0;line-height:0;vertical-align:top}#fancybox-frame{width:100%;height:100%;border:0;display:block}#fancybox-left,#fancybox-right{position:absolute;bottom:0;height:100%;width:35%;cursor:pointer;outline:0;background:transparent url('../../../img/lib/fancybox/blank.gif');z-index:1102;display:none}#fancybox-left{left:0}#fancybox-right{right:0}#fancybox-left-ico,#fancybox-right-ico{position:absolute;top:50%;left:-9999px;width:30px;height:30px;margin-top:-15px;cursor:pointer;z-index:1102;display:block}#fancybox-left-ico{background-image:url('../../../img/lib/fancybox/fancybox.png');background-position:-40px -30px}#fancybox-right-ico{background-image:url('../../../img/lib/fancybox/fancybox.png');background-position:-40px -60px}#fancybox-left:hover,#fancybox-right:hover{visibility:visible}#fancybox-left:hover span{left:20px}#fancybox-right:hover span{left:auto;right:20px}.fancybox-bg{position:absolute;padding:0;margin:0;border:0;width:20px;height:20px;z-index:1001}#fancybox-bg-n{top:-20px;left:0;width:100%;background-image:url('../../../img/lib/fancybox/fancybox-x.png')}#fancybox-bg-ne{top:-20px;right:-20px;background-image:url('../../../img/lib/fancybox/fancybox.png');background-position:-40px -162px}#fancybox-bg-e{top:0;right:-20px;height:100%;background-image:url('../../../img/lib/fancybox/fancybox-y.png');background-position:-20px 0}#fancybox-bg-se{bottom:-20px;right:-20px;background-image:url('../../../img/lib/fancybox/fancybox.png');background-position:-40px -182px}#fancybox-bg-s{bottom:-20px;left:0;width:100%;background-image:url('../../../img/lib/fancybox/fancybox-x.png');background-position:0 -20px}#fancybox-bg-sw{bottom:-20px;left:-20px;background-image:url('../../../img/lib/fancybox/fancybox.png');background-position:-40px -142px}#fancybox-bg-w{top:0;left:-20px;height:100%;background-image:url('../../../img/lib/fancybox/fancybox-y.png')}#fancybox-bg-nw{top:-20px;left:-20px;background-image:url('../../../img/lib/fancybox/fancybox.png');background-position:-40px -122px}#fancybox-title{font-family:Helvetica;font-size:12px;z-index:1102}.fancybox-title-inside{padding-bottom:10px;text-align:center;color:#333;background:#fff;position:relative}.fancybox-title-outside{padding-top:10px;color:#fff}.fancybox-title-over{position:absolute;bottom:0;left:0;color:#FFF;text-align:left}#fancybox-title-over{padding:10px;background-image:url('../../../img/lib/fancybox/fancy_title_over.png');display:block}.fancybox-title-float{position:absolute;left:0;bottom:-20px;height:32px}#fancybox-title-float-wrap{border:0;border-collapse:collapse;width:auto}#fancybox-title-float-wrap td{border:0;white-space:nowrap}#fancybox-title-float-left{padding:0 0 0 15px;background:url('../../../img/lib/fancybox/fancybox.png') -40px -90px no-repeat}#fancybox-title-float-main{color:#FFF;line-height:29px;font-weight:bold;padding:0 0 3px 0;background:url('../../../img/lib/fancybox/fancybox-x.png') 0 -40px}#fancybox-title-float-right{padding:0 0 0 15px;background:url('../../../img/lib/fancybox/fancybox.png') -55px -90px no-repeat}.fancybox-ie6 #fancybox-close{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_close.png',sizingMethod='scale')}.fancybox-ie6 #fancybox-left-ico{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_nav_left.png',sizingMethod='scale')}.fancybox-ie6 #fancybox-right-ico{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_nav_right.png',sizingMethod='scale')}.fancybox-ie6 #fancybox-title-over{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_title_over.png',sizingMethod='scale');zoom:1}.fancybox-ie6 #fancybox-title-float-left{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_title_left.png',sizingMethod='scale')}.fancybox-ie6 #fancybox-title-float-main{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_title_main.png',sizingMethod='scale')}.fancybox-ie6 #fancybox-title-float-right{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_title_right.png',sizingMethod='scale')}.fancybox-ie6 #fancybox-bg-w,.fancybox-ie6 #fancybox-bg-e,.fancybox-ie6 #fancybox-left,.fancybox-ie6 #fancybox-right,#fancybox-hide-sel-frame{height:expression(this.parentNode.clientHeight+"px")}#fancybox-loading.fancybox-ie6{position:absolute;margin-top:0;top:expression((-20+(document.documentElement.clientHeight ? document.documentElement.clientHeight/2:document.body.clientHeight/2)+(ignoreMe = document.documentElement.scrollTop ? document.documentElement.scrollTop:document.body.scrollTop))+'px')}#fancybox-loading.fancybox-ie6 div{background:transparent;filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_loading.png',sizingMethod='scale')}.fancybox-ie .fancybox-bg{background:transparent!important}.fancybox-ie #fancybox-bg-n{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_n.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-ne{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_ne.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-e{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_e.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-se{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_se.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-s{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_s.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-sw{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_sw.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-w{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_w.png',sizingMethod='scale')}.fancybox-ie #fancybox-bg-nw{filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='../../../img/lib/fancybox/fancy_shadow_nw.png',sizingMethod='scale')}@font-face{font-family:Inconsolata;src:url(../fonts/Inconsolata.otf)}.clear{overflow:hidden;width:100%}a.button{background:transparent url(../img/v2/bg_button_a.gif) no-repeat scroll top right;color:#444;display:block;float:right;font:normal 12px arial,sans-serif;height:24px;margin-right:6px;padding-right:18px;text-decoration:none}a.button span{background:transparent url(../img/v2/bg_button_span.gif) no-repeat;display:block;line-height:14px;letter-spacing:1px;padding:5px 0 5px 18px}a.button:active{background-position:bottom right;outline:0}a.button:active span{background-position:bottom left;padding:6px 0 4px 18px}#plans h2{margin-bottom:10px;padding-bottom:10px}#plans h3{margin:20px 0 10px;font-size:15px}#featuretour .hero-shot{margin:10px 0 40px;width:100%;height:216px;background:url(../img/featuretour/heroshot.jpg) no-repeat;color:#c5efff}#featuretour .hero-shot h1{position:absolute;left:30px;bottom:30px;font-weight:normal;font-size:4.5em;line-height:1.1;color:#f5fba5}#featuretour div.center{margin:20px 0;width:100%;text-align:center}#featuretour a.btn{width:250px;-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px;border:1px solid #0f2557;background:#245192;background:-webkit-gradient(linear,0 0,0 100%,color-stop(0,#245192),color-stop(0.75,#1e3b73),color-stop(1,#12295d));background:-moz-linear-gradient(#245192,#1e3b73 75%,#12295d);padding:10px 75px 10px;font-weight:bold;font-size:24px;font-family:Arial;text-decoration:none;color:#fff;text-shadow:-1px -1px 2px #465f97}#featuretour a.btn:hover{border:1px solid #19377c;background:#3567b7;background:-moz-linear-gradient(#3567b7,#294f99 75%,#1e3e86);background:-webkit-gradient(linear,0 0,0 100%,color-stop(0,#3567b7),color-stop(0.75,#294f99),color-stop(1,#1e3e86));text-shadow:-1px -1px 2px #465f97}#featuretour .feature{float:left;width:47%;padding:0 20px 40px 0;vertical-align:top}#featuretour h4{margin:0 0 10px;padding:0}#featuretour img{float:left;margin-right:10px;width:200px;border:1px solid #999}#featuretour p{display:block;margin:0 0 10px}#featuretour .example{font-size:90%}#featuretour .compoundheadline h1{float:left}#featuretour .compoundheadline:after{display:block;clear:both;height:0;content:".";visibility:hidden}#featuretour .compoundheadline p{position:relative;left:5px;top:7px;color:#777}.markItUp *{margin:0;padding:0}.markItUp a:link,.markItUp a:visited{color:#000;text-decoration:none}.markItUp{margin:5px 0}.markItUpContainer{font:11px Verdana,Arial,Helvetica,sans-serif}.markItUpEditor{font:12px 'Courier New',Courier,monospace;padding:5px;width:690px;height:320px;clear:both;line-height:18px;overflow:auto}.markItUpPreviewFrame{overflow:auto;background-color:#FFF;width:99.9%;height:300px;margin:5px 0}.markItUpFooter{width:100%}.markItUpResizeHandle{overflow:hidden;width:22px;height:5px;margin-left:auto;margin-right:auto;background-image:url(images/handle.png);cursor:n-resize}.markItUp .markItUpHeader ul{padding:0}.markItUp .markItUpHeader ul li{margin-top:0;margin-bottom:0;list-style:none;float:left;position:relative}.markItUpHeader ul li:before{content:""}.markItUpHeader ul li:hover>ul{display:block}.markItUpHeader ul .markItUpDropMenu{background:transparent url(images/menu.png) no-repeat 115% 50%;margin-right:5px}.markItUpHeader ul .markItUpDropMenu li{margin-right:0}.markItUpHeader ul ul{display:none;position:absolute;top:18px;left:0;background:#FFF;border:1px solid #000}.markItUpHeader ul ul li{float:none;border-bottom:1px solid #000}.markItUpHeader ul ul .markItUpDropMenu{background:#FFF url(images/submenu.png) no-repeat 100% 50%}.markItUpHeader ul .markItUpSeparator{margin:0 10px;width:1px;height:16px;overflow:hidden;background-color:#CCC}.markItUpHeader ul ul .markItUpSeparator{width:auto;height:1px;margin:0}.markItUpHeader ul ul ul{position:absolute;top:-1px;left:150px}.markItUpHeader ul ul ul li{float:none}.markItUpHeader ul a{display:block;width:16px;height:16px;text-indent:-10000px;background-repeat:no-repeat;padding:3px;margin:0}.markItUpHeader ul ul a{display:block;padding-left:0;text-indent:0;width:120px;padding:5px 5px 5px 25px;background-position:2px 50%}.markItUpHeader ul ul a:hover{color:#FFF;background-color:#000}.markItUp .markItUpButton1 a{background-image:url()}.markItUp .markItUpButton2 a{background-image:url()}.markItUp .markItUpButton3 a{background-image:url()}.markItUp .markItUpButton4 a{background-image:url()}.markItUp .markItUpButton5 a{background-image:url()}.markItUp .markItUpButton6 a{background-image:url()}.markItUp .markItUpButton7 a{background-image:url()}.markItUp .markItUpButton8 a{background-image:url()}.markItUp .markItUpButton9 a{background-image:url()}.markItUp .markItUpButton10 a{background-image:url()}.markItUp .markItUpButton11 a{background-image:url()}.markItUp .markItUpButton12 a{background-image:url()}.markItUp .markItUpButton13 a{background-image:url()}.markItUp .markItUpButton14 a{background-image:url()}.markItUp .preview a{background-image:url()}.patch-container{float:left;width:100%}.diff-container{margin:0 auto 30px;border:1px solid #ddd;width:100%}.patch-container .heading>h3{padding:10px 0 0;font-weight:normal;font-size:18px;line-height:2;color:#666}#content .patch-container .heading>h3{color:#666}.patch-container .heading>h3>a{color:inherit}.patch-container .heading.conflicts>h3:after{content:"\0020(conflicts)";color:#c00}#compare .patch-container .heading a[href="#files"],.patch-container .heading a[href="#files"]{position:absolute;right:0;top:0;width:15px;height:15px;background:url(../img/up-arrow-sprite.png);text-indent:-9999px}#compare .patch-container .heading a[href="#files"]:focus,#compare .patch-container .heading a[href="#files"]:hover,.patch-container .heading a[href="#files"]:focus,.patch-container .heading a[href="#files"]:hover{background-position:-15px}.diff-container .box-container,.diff-container .scrollable,.diff-container .from-numbers-box,.diff-container .to-numbers-box,.diff-container .from-revision-blame-box,.diff-container .to-revision-blame-box,.diff-container .from-auth-blame-box,.diff-container .to-auth-blame-box,.diff-container .segment-box{position:relative}.diff-container .box-container,.diff-container .box,.diff-container .numbers-box,.diff-container .segment-box{height:100%}.diff-container .box-container{border-left:none;float:left}.diff-container .box{width:100%}.diff-container .scrollable{overflow-x:auto;overflow-y:auto}.diff-container.unified .scrollable,.diff-container.view-annotation .scrollable{overflow-y:hidden}.diff-container .from-box{float:left;background:transparent url(../images/spinner-sbs-diff.gif) fixed no-repeat 21% 50%}.diff-container.unified .from-box{overflow:hidden}.diff-container .to-box{float:left;background:transparent url(../images/spinner-sbs-diff.gif) fixed no-repeat 77% 50%}.diff-container.unified .to-box{background-position:50% 50%}.diff-container .from-box.all-lines-visible,.diff-container .to-box.all-lines-visible{background-image:none}.diff-container .to-numbers-box{border-left:none;border-right:1px solid #ccc}.diff-container .to-box{border-left:none}.diff-container .from-numbers-box,.diff-container .to-numbers-box{float:left;text-align:right;overflow:hidden}.diff-container .segment-box{border-left:1px solid #ccc;border-right:1px solid #ccc}.diff-container>div{border-left:1px solid #ccc;border-right:0}.diff-container.unified .from-numbers-box,.diff-container .from-box,.diff-container .heading{border-left:none}.diff-container .from-revision-blame-box,.diff-container .to-revision-blame-box,.diff-container .from-auth-blame-box,.diff-container .to-auth-blame-box{float:left;overflow:hidden;color:#036}.diff-container .gutter-box{float:left;overflow:hidden}.diff-container .segment-box{float:left;width:30px;overflow:hidden}.diff-container .content-box{position:relative}.diff-container .page{background:#fff;margin:0;padding:0;position:absolute}.diff-container pre{-webkit-border-radius:0;-moz-border-radius:0;border-radius:0;border:0;background:inherit;white-space:pre}.diff-container pre{font-family:"Bitstream Vera Sans Mono","DejaVu Sans Mono",Monaco,monospace;font-size:11px;margin:0;height:16px;line-height:16px;padding:0 .2em;border-width:1px 0;border-style:solid;border-color:transparent}.diff-container a{color:#999;text-decoration:none}.diff-container a:hover,.diff-container a:active{color:#00e;text-decoration:underline}.diff-container .line-numbers{left:0;width:100%}.diff-container .addition,.diff-container .change.to,.diff-container .ediffChangedB{background-color:#cfc;border-color:#cfc}.diff-container ins{background-color:#cfc;text-decoration:none;display:inline-block}.diff-container .change.to ins{background-color:#9f9}.diff-container .anchored-line .ediffChangedB{background-color:#e5ffd6;border-color:#e5ffd6}.diff-container .deletion,.diff-container .change.from,.diff-container .ediffChangedA{background-color:pink;border-color:pink}.diff-container .addition.conflict,.diff-container .change.conflict{background-color:#ffffe0;border-color:#ffffe0}.diff-container del{background-color:pink;text-decoration:none;display:inline-block}.diff-container .change.from del{background-color:#ff909b}.diff-container .anchored-line .ediffChangedA{background-color:#ffdfd6;border-color:#ffdfd6}.diff-container .change{background-color:#f4f4f4;border-color:#f4f4f4}.diff-container .anchored-line{background-color:#ffffe0;border-color:#ffffe0}.diff-container .ediffChangedA,.diff-container .ediffChangedB{border-width:1px 0;border-style:solid}.diff-container .start .ediffChangedA,.diff-container .start .ediffChangedB{border-top:0}.diff-container .end .ediffChangedA,.diff-container .end .ediffChangedB{border-bottom:0}.diff-container .from-box .change.conflict.start,.diff-container .from-numbers-box .change.conflict.start,.diff-container .addition.conflict.start.end,.diff-container .change.from.start,.diff-container .start{border-top-color:#ccc}.diff-container .addition.conflict.end,.diff-container .change.to.end,.diff-container .end{border-bottom-color:#ccc}.diff-container.unified .addition.conflict.end,.diff-container .addition.conflict.start.end{border-bottom-color:#ffffe0}.diff-container.unified .from-numbers-box .change.conflict.start,.diff-container .conflict.from.start,.diff-container .conflict.to.end{border-color:#ffffe0}.diff-container .skipped,.diff-container .skipped-top,.diff-container .skipped-bottom{border-color:#fff;background:#ebf1fb url('../images/diff-skipped-background.png') repeat-x bottom left;padding:0}.diff-container .skipped-top{border-top-color:#ebf1fb}.diff-container .skipped-bottom{background-image:none}.diff-container .skipped span,.diff-container .skipped-bottom span{display:block;background:transparent url('../images/diff-skipped-background.png') repeat-x top left}.diff-container .skipped-top span{visibility:hidden}.diff-container .addition.conflict.start,.diff-container .addition.conflict.start.end,.diff-container .addition.conflict.end{color:#999;border-color:#ccc;background-color:#fffacd}.diff-container>.overlay{width:100%;height:100%;background:transparent;border:0;cursor:pointer}.diff-container.unified>.overlay,.diff-container.unified .overlay-x,.diff-container.unified .overlay-y,.diff-container.scrollable-on>.overlay{display:none}.diff-container.scrollable-off .overlay-x,.diff-container.scrollable-off .overlay-y{position:absolute;background:#fff;-ms-filter:"alpha(opacity=75)";filter:alpha(opacity=75);-moz-opacity:.75;opacity:.75}.diff-container.scrollable-off .overlay-x{left:0;right:15px;bottom:0}.diff-container.scrollable-off .overlay-y{top:0;right:0;bottom:0}.diff-container .margin-buttons{position:absolute;right:0}.diff-container .margin-buttons a{display:block;text-indent:-10000em;position:absolute;left:2px;width:9px;height:15px}.diff-container .margin-buttons .prev-hunk,.diff-container .margin-buttons .next-hunk{background:#fff url(../img/sbs-diff-sprite.png) no-repeat -3px -15px}.diff-container .margin-buttons .prev-hunk:hover,.diff-container .margin-buttons .prev-hunk:active{background-position:-18px -15px}.diff-container .margin-buttons .prev-hunk{left:12px}.diff-container .margin-buttons .next-hunk{background-position:-3px -30px}.diff-container .margin-buttons .next-hunk:hover,.diff-container .margin-buttons .next-hunk:active{background-position:-18px -30px}.diff-container .margin-buttons .more-context{width:15px;background:#fff url(../img/sbs-diff-sprite.png) no-repeat 0 0}.diff-container .margin-buttons .more-context:hover,.diff-container .margin-buttons .more-context:active{background-position:-15px 0}.c{color:#998;font-style:italic}.err{color:#a61717;background-color:#e3d2d2}.cm{color:#998;font-style:italic}.c1{color:#998;font-style:italic}.cs{color:#999;font-style:italic}.gd{color:#000;background-color:#fdd}.ge{font-style:italic}.gr{color:#a00}.gh{color:#999}.gi{color:#000;background-color:#dfd}.go{color:#888}.gp{color:#555}.gu{color:#aaa}.gt{color:#a00}.k{color:#004080}.kd{color:#004080}.kt{color:#458}.m{color:#099}.s{color:#b84}.na{color:#008080}.nb{color:#999}.nc{color:#458}.no{color:#008080}.ni{color:#800080}.ne{color:#900}.nf{color:#900}.nn{color:#555}.nt{color:#000080}.nv{color:#008080}.w{color:#bbb}.mf{color:#099}.mh{color:#099}.mi{color:#099}.mo{color:#099}.sb{color:#b84}.sc{color:#b84}.sd{color:#b84}.s2{color:#b84}.se{color:#b84}.sh{color:#b84}.si{color:#b84}.sx{color:#b84}.sr{color:#808000}.s1{color:#b84}.ss{color:#b84}.bp{color:#999}.vc{color:#008080}.vg{color:#008080}.vi{color:#008080}.il{color:#099}cssutils-1.0/sheets/096.css0000666000175000017500000001023112126054542014236 0ustar hugohugo/* default stylesheet for all variations */ .jsonly { display: none } .stylenav { text-align: right; margin-top: -1em } html, body { padding: 0; margin: 0 } body { font: normal 90%/1.1 Georgia, Verdana, "Lucida Grande", Helvetica, sans-serif; color: #fff; background-color: #344 } h1, h2, h3, h4, h5, h6 { font: normal 2.4em Verdana, "Lucida Grande", Arial, Helvetica, sans-serif } h1, h2, h3, caption { color: #a00; margin: 0.3em -0.4em 0.5em -0.5em } h1, h2 { letter-spacing: -0.05em } h2, h3, caption { margin: 0.3em -1.3em 0.3em 0 } h2 { font-size: 1.6em; border-right: 1.3em solid #677; border-bottom: 1px dotted #677; padding-left: 0.1em; padding-bottom: 0.1em; margin-top: 2.1em; margin-bottom: 0.4em } h3 { font-size: 0.9em; text-transform: uppercase; margin-top: 1.2em; margin-bottom: 0.1em } caption { font-size: 1.05em; text-align: left; margin-bottom: 0.2em } h4 { font-size: 0.9em; font-weight: bold; margin: 1.2em 0 0 } h5, h6 { font-size: 1em; margin: 0 } h6 { font-size: 0.9em } p, ol, ul, dl { line-height: 1.3; margin-top: 0; margin-bottom: 1em } ul { list-style-type: square } ul.code { line-height: 1.3 } li, dd { margin-bottom: 0.3em } dt { font-weight: bold } pre, code { color: #00a; line-height: 1.4; font-size: 1.1em } pre { border: 1px solid #eee; overflow: auto } table code, table pre { font-size: 1.3em } .deprecated { color: #888 } table { font-size: 0.9em; border-collapse: collapse } tr { vertical-align: top; line-height: 1.3 } td, th { text-align: left; padding: 0.4em 0.5em; border-bottom: 1px dotted #667 } td.center { text-align: center } tr:hover, li:hover { background-color: #f8f8f8 } acronym, .explain { border-bottom: 1px dotted #344 } a { text-decoration: none; color: #fff; border-bottom: 1px solid #aaa } #main a { color: #a00 } a:visited { color: #eee } #main a:visited { color: #344 } a:hover { text-decoration: underline; color: #fff } #main a:hover { background-color: #f5f8ff } #main a:active { color: #fff; background-color: #abb } label { display: block; padding: 0.5em 0 0.1em } input, textarea { font: bold 1em Georgia, Verdana, "Lucida Grande", Helvetica, sans-serif; background-color: #eee; width: 100%; border: 1px inset } #submit, textarea { margin-bottom: 1.5em } #submit { font-weight: bold; color: #00a; background-color: #fff; border: 1px outset; margin-top: 2em } input:focus, input:hover, textarea:focus, textarea:hover { font-weight: bold; background-color: #fff; border-style: solid } #submit:hover, #submit:focus { background-color: #eee; border-style: solid } #submit:active { border-style: inset } #header { padding: 1.1em 5% 0; padding: 40px 5% 0; color: #334; background: #fff url(../static/img/header_r.jpg) no-repeat; border-bottom: 1px solid #344; height: 90px; he\ight: 50px } #header dt, #header p { font-family: Arial, Helvetica, sans-serif; letter-spacing: 0.3em } #header a { color: #334 } #main .nav { padding-bottom: 0.5em; border-bottom: 3px solid #eee; margin-bottom: 1em; margin-left: -8% } .nav dt, .nav dd, .nav dd ul, .nav li { display: inline } .nav dt { font-weight: bold } .nav li { font-weight: bold; padding-right: 0.5em } .nav a { font-weight: normal } #footer { padding: 1em 5% 1em 10%; border-top: 3px double #fff; text-align: right } #main { color: #000; background-color: #fff; padding: 1em 26% 1em 12% } .ext { margin-top: 2em } .ext a { font-size: 0.8em }cssutils-1.0/sheets/hacks.css0000666000175000017500000000133512126054542015016 0ustar hugohugo.normal {background-color: gray;} .backslash {bac\kground-color: gray;} html>body .childselector {background-color: green;} html>/**/body .childselector-with-comment {background-color: orange} .colon-default2, x:default html>/**/body .colon-default, x:default *:not(hr) .not-hr {background-color: red;} * html .ie-only-1 {background-color: blue;} *+html .ie-only-2 {background-color: blue;} *+html .ie-only-3 {background-color: blue;} html:first-child .first-child-2 {background-color: red;} /* does not work as CSSUnknownRule read: @mediaall { .mediaall { background-color: red; }} */ .not-class:not([class='XXX']) {background-color: red;} @media all and (min-width: 0) { .mediaquery { background-color: red;} } cssutils-1.0/sheets/u_simple.css0000666000175000017500000000045012126054542015537 0ustar hugohugo@media screen { h1 { foo: bar } h2 { thing: 1; whatsit: 2px } h3 { foo: 2 ! important } } foo :lang(c) #bob { a: 1; b: 2; } foo { } cssutils-1.0/sheets/vars.css0000666000175000017500000000033612126054542014700 0ustar hugohugo@import "varsimport.css"; @variables { c2: "c2 own file (OVERWRITTEN)"; c3: "c1 own file" } a1 { content: var(c1); } a2 { content: var(c2); } a3 { content: var(c3); } cssutils-1.0/sheets/xhtml2.css0000666000175000017500000001145612126054542015150 0ustar hugohugo@namespace url("http://www.w3.org/2002/06/xhtml2/"); /* A sample style sheet for XHTML 2.0 This style sheet describes a very incomplete, sample rendering of XHTML 2.0 elements. Editor: Masayasu Ishikawa Revision: $Id$ */ /* new elements */ section, h, nl, label, l, blockcode, separator, di { display: block; } section, h, nl, label, l, blockcode, di { unicode-bidi: embed } nl { margin: 1.33em 0 } summary, standby, handler { display: none } blockcode { font-family: monospace; white-space: pre } separator { border-bottom: thin black solid; border: 1px; inset; width 100%} h { display: block; font-weight: bolder; font-family: sans-serif } h1, h2, h3, h4, h5, h6 { font-family: sans-serif; font-weight: bolder } body h, h1 { font-size: 2em; margin: .67em 0; } section h, h2 { font-size: 1.5em; margin: .83em 0; } section section h, h3 { font-size: 1.17em; margin: 1em 0; } section section section h, h4, p, blockquote, ul, ol, dl { margin: 1.33em 0; } section section section section h, h5 { font-size: .83em; line-height: 1.17em; margin: 1.67em 0; } section section section section section h, h6 { font-size: .67em; margin: 2.33em 0; } *[edit="deleted"] { display: none } /* no special presentation by default *[edit="inserted"] { } *[edit="changed"] { } *[edit="moved"] { } */ /* experimental navigation list style */ nl { height: 1.5em; overflow: hidden; margin: 0; line-height: normal !important; white-space: nowrap; text-align: start; cursor: default; border-width: 2px !important; border-style: inset !important; vertical-align: baseline; padding: 0; } nl:hover { height: auto; overflow: visible; } nl > li, nl > label { display: block; min-height: 1em; line-height: normal !important; } nl > li, nl > label { padding: 0 5px 0 3px; } nl > li { margin-left: 1em; } nl > label { font-weight: bold; } nl > nl > label { display: block; line-height: normal !important; font-style: italic; font-weight: bold; } nl > nl > li { padding-left: 2em; font-style: normal; font-weight: normal; } /* inherited elements */ html, body, div, p, h1, h2, h3, h4, h5, h6, address, blockquote, pre, ol, ul, dl, dt, dd { display: block } li { display: list-item } head, style, link, meta { display: none } table { display: table; border-spacing: 0; border-top: thin black solid; border-left: thin black solid } tr { display: table-row } thead { display: table-header-group } tbody { display: table-row-group } tfoot { display: table-footer-group } col { display: table-column } colgroup { display: table-column-group } td, th { display: table-cell; border-right: thin black solid; border-bottom: thin black solid; padding 2px } caption { display: table-caption } table:hover summary { display: block } th { font-weight: bolder; text-align: center } caption { text-align: center } body { padding: 8px; line-height: 1.2 } strong { font-weight: bolder } blockquote { margin-left: 4em; margin-right: 4em } cite, em, q, var, address { font-style: italic } pre code, kbd, samp { font-family: monospace } pre { white-space: pre } sub, sup { font-size: smaller } sub { vertical-align: sub } sup { vertical-align: super } ol, ul, dd { margin-left: 4em } ol { list-style-type: decimal } ol ul, ul ol, ul ul, ol ol { margin-top: 0; margin-bottom: 0 } abbr[title] { border-bottom: dotted 1px } :link { text-decoration: underline; color: blue; } :focus { outline: thin dotted invert } /* Hover effects should be default */ :link:hover,:link:visited { color: #b7f } /* begin bidirectionality settings (do not change) */ *[dir="ltr"] { direction: ltr; unicode-bidi: embed } *[dir="rtl"] { direction: rtl; unicode-bidi: embed } *[dir="lro"] { direction: ltr; unicode-bidi: bidi-override } *[dir="rlo"] { direction: rtl; unicode-bidi: bidi-override } /* block-level elements */ body, div, p, hr, h1, h2, h3, h4, h5, h6, address, blockquote, pre, ol, ul, li, di, dt, dd, table, thead, tbody, tfoot, tr, td, th, col, colgroup, caption, object, summary, standby, blockcode { unicode-bidi: embed } /* end bidi settings */ /* end xhtml2.css */cssutils-1.0/sheets/097.css0000666000175000017500000001023112126054542014237 0ustar hugohugo/* default stylesheet for all variations */ .jsonly { display: none } .stylenav { text-align: right; margin-top: -1em } html, body { padding: 0; margin: 0 } body { font: normal 90%/1.1 Georgia, Verdana, "Lucida Grande", Helvetica, sans-serif; color: #fff; background-color: #344 } h1, h2, h3, h4, h5, h6 { font: normal 2.4em Verdana, "Lucida Grande", Arial, Helvetica, sans-serif } h1, h2, h3, caption { color: #a00; margin: 0.3em -0.4em 0.5em -0.5em } h1, h2 { letter-spacing: -0.05em } h2, h3, caption { margin: 0.3em -1.3em 0.3em 0 } h2 { font-size: 1.6em; border-right: 1.3em solid #677; border-bottom: 1px dotted #677; padding-left: 0.1em; padding-bottom: 0.1em; margin-top: 2.1em; margin-bottom: 0.4em } h3 { font-size: 0.9em; text-transform: uppercase; margin-top: 1.2em; margin-bottom: 0.1em } caption { font-size: 1.05em; text-align: left; margin-bottom: 0.2em } h4 { font-size: 0.9em; font-weight: bold; margin: 1.2em 0 0 } h5, h6 { font-size: 1em; margin: 0 } h6 { font-size: 0.9em } p, ol, ul, dl { line-height: 1.3; margin-top: 0; margin-bottom: 1em } ul { list-style-type: square } ul.code { line-height: 1.3 } li, dd { margin-bottom: 0.3em } dt { font-weight: bold } pre, code { color: #00a; line-height: 1.4; font-size: 1.1em } pre { border: 1px solid #eee; overflow: auto } table code, table pre { font-size: 1.3em } .deprecated { color: #888 } table { font-size: 0.9em; border-collapse: collapse } tr { vertical-align: top; line-height: 1.3 } td, th { text-align: left; padding: 0.4em 0.5em; border-bottom: 1px dotted #667 } td.center { text-align: center } tr:hover, li:hover { background-color: #f8f8f8 } acronym, .explain { border-bottom: 1px dotted #344 } a { text-decoration: none; color: #fff; border-bottom: 1px solid #aaa } #main a { color: #a00 } a:visited { color: #eee } #main a:visited { color: #344 } a:hover { text-decoration: underline; color: #fff } #main a:hover { background-color: #f5f8ff } #main a:active { color: #fff; background-color: #abb } label { display: block; padding: 0.5em 0 0.1em } input, textarea { font: bold 1em Georgia, Verdana, "Lucida Grande", Helvetica, sans-serif; background-color: #eee; width: 100%; border: 1px inset } #submit, textarea { margin-bottom: 1.5em } #submit { font-weight: bold; color: #00a; background-color: #fff; border: 1px outset; margin-top: 2em } input:focus, input:hover, textarea:focus, textarea:hover { font-weight: bold; background-color: #fff; border-style: solid } #submit:hover, #submit:focus { background-color: #eee; border-style: solid } #submit:active { border-style: inset } #header { padding: 1.1em 5% 0; padding: 40px 5% 0; color: #334; background: #fff url(../static/img/header_r.jpg) no-repeat; border-bottom: 1px solid #344; height: 90px; he\ight: 50px } #header dt, #header p { font-family: Arial, Helvetica, sans-serif; letter-spacing: 0.3em } #header a { color: #334 } #main .nav { padding-bottom: 0.5em; border-bottom: 3px solid #eee; margin-bottom: 1em; margin-left: -8% } .nav dt, .nav dd, .nav dd ul, .nav li { display: inline } .nav dt { font-weight: bold } .nav li { font-weight: bold; padding-right: 0.5em } .nav a { font-weight: normal } #footer { padding: 1em 5% 1em 10%; border-top: 3px double #fff; text-align: right } #main { color: #000; background-color: #fff; padding: 1em 26% 1em 12% } .ext { margin-top: 2em } .ext a { font-size: 0.8em }cssutils-1.0/sheets/sample_7.css0000666000175000017500000001675612126054542015451 0ustar hugohugo/* CSS Document */ body { background:url(image/bg.gif) 0 0 repeat; /*background-color:#D1CFD0;*/ padding:0px; margin:0px; } /*table.outer-table{ background:url(image/bg.gif) 0 0 repeat; width:1024px; border:0; }*/ td a{margin:0; padding:0;} table.inner-table{ background:url(image/background-inner.jpg) 0 0 repeat-y; width:1004px; margin:0 10px; border:0; } p,img, input,span { padding:0px; margin:0px; } ul{ list-style-type:none; padding:0px; margin:0px; } .signup { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#FF8619; text-decoration:none; font-weight:bold; padding:0px 5px 0px 0px; text-decoration:underline; } .signup a{ font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#FF8619; text-decoration:none; font-weight:bold; padding:0px 5px 0px 0px; } .signup a:hover{ font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#000066; text-decoration:none; font-weight:bold; padding:0px 5px 0px 0px; } .rull { background-image:url(image/rull.jpg); background-repeat:no-repeat; } p.textlink { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:21px; color:#605E5F; font-weight:normal; padding:36px 0px 0px 2px; } p.textlink a{ font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:21px; color:#605E5F; text-decoration:none; } p.textlink a:hover { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; color:#FF8619; } .text_welcome { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:20px; color:#FF8619; text-decoration:none; font-weight:normal; } .text_welcome1 { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:20px; color:#3B65A5; text-decoration:none; font-weight:normal; } .soliborder{ border-top: solid #BFC8CF 1px; } .padding { padding-top:21px; padding-left:17px; padding-bottom:12px; } .text { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#000000; font-weight:normal; } .texte { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#0E4190; text-decoration:underline; font-weight:normal; } .texte a { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#0E4190; text-decoration:underline; font-weight:normal; } .texte:hover { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#FF8619; text-decoration:none; font-weight:normal; } .text3 { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#605E5F; text-decoration:none; font-weight:normal; } .text3{ list-style:inside disc; padding:0px 0px 5px 5px; margin-left:10px; } .text4 { list-style:circle inside; margin:5px 0px 0px 15px; } .text_orange { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#FF8619; text-decoration:none; font-weight:bold; } .bacground_map { background-image:url(image/map.jpg); background-repeat:no-repeat; } .padding2 { padding:8px 35px 118px 17px; } .text_2 { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#605E5F; text-decoration:none; font-weight:normal; } .text_orange2 { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#FF8619; text-decoration:none; font-weight:bold; } .bacground_map2 { background-image:url(image/map.jpg); background-repeat:no-repeat; } .padding3 { padding-top:23px 0px 8px 14px; } .text_blue { padding:23px 0px 8px 14px; font-family:Arial, Helvetica, sans-serif; font-size:12px; line-height:normal; color:#0E4190; text-decoration:none; font-weight:bold; } .text_blue a { padding:23px 0px 8px 14px; font-family:Arial, Helvetica, sans-serif; font-size:12px; line-height:normal; color:#0E4190; text-decoration:none; font-weight:bold; margin:0; padding:0; } .text_blue a:hover { padding:23px 0px 8px 14px; font-family:Arial, Helvetica, sans-serif; font-size:12px; line-height:normal; color:#0E4190; text-decoration:none; font-weight:bold; margin:0; padding:0; } /*.text_blue a:hover { padding:23px 0px 8px 14px; font-family:Arial, Helvetica, sans-serif; font-size:12px; line-height:normal; color:#0E4190; text-decoration:none; font-weight:bold; }*/ .menuh { height:30px; padding-top:9px; } .text_blue2 a{ font-family:Arial, Helvetica, sans-serif; font-size:12px; line-height:normal; color:#0E4190; text-decoration:none; font-weight:bold; } .text_blue2 a:hover{ font-family:Arial, Helvetica, sans-serif; font-size:12px; line-height:normal; color:#FF6600; text-decoration:underline; font-weight:bold; } .rightpannel_bg{ background-image:url(image/right_bg.gif); background-position:bottom; vertical-align:bottom; background-repeat:repeat-x; border:solid #E2964B 1px; background-color:#FFF; padding:5px; } .rightpannel_bg_last{ border-left:solid #BFC8CF 1px; border-right:solid #BFC8CF 1px; border-bottom:solid #BFC8CF 1px; background-color:#ffffff; padding:5px; } /*------------------------Button----------------------*/ .sup { float:right; margin:5px 0 0 0 } /*------------------------Button End-------------------*/ ul.link{ list-style-type:none; padding:12px 0px 10px 0px; } ul.link li{ background:url(image/icon.gif) no-repeat 0 5px; font:normal 12px/19px Arial, Helvetica, sans-serif "Trebuchet MS"; color:#605E5F; margin:0px 2px 0px 12px; } ul.link li span{ font:normal 12px/19px Arial, Helvetica, sans-serif "Trebuchet MS"; color:#605E5F; margin:0px 0px 0px 8px; } ul.link li span.align{ font:normal 12px/19px Arial, Helvetica, sans-serif "Trebuchet MS"; color:#605E5F; margin:0px 0px 0px 8px; } /*-----------------------------footer----------------------*/ .boackground_footr { background-image:url(image/footer-rull.gif); background-repeat:no-repeat; } .textsky { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:19px; color:#1B57B3; font-weight:normal; text-decoration:none; } .textsky span{padding:0 5px;} .textsky:hover { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:19px; color:#FF8619; font-weight:normal; text-decoration:none; } .text_ash { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:11px; line-height:19px; color:#605E5F; font-weight:normal; } .style1 {color: #FF9900} .text_3 { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#605E5F; text-decoration:none; font-weight:normal; padding: 0 0 12px 0; } .text-as { font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#000000; font-weight:normal; padding: 8px 15px 5px 8px; } .text-as1 { background:url(image/bullet.gif) left 10px no-repeat; font-family:Arial, Helvetica, sans-serif "Trebuchet MS"; font-size:12px; line-height:16px; color:#000000; font-weight:normal; padding: 8px 15px 5px 15px; } cssutils-1.0/sheets/slashcode.css0000666000175000017500000004432512126054542015700 0ustar hugohugobody, div, form, #links ul li, #links ul, #topnav, img { font-size: 100%; } body { background: #fff; /* font: 85%/150% verdana, sans-serif; */ font: 84%/150% sans-serif; } * a { color: #006699; } a:visited { color: #6a6a6a; } #topnav { position: relative; height: 55px; margin: 5px 0 0 0; background: #f57300 url(/images/topnav-orange-bg.png) repeat-x left; clear: both; } #topnav div.search { position: absolute; top: 10px; right: 1em; text-align: right; white-space: nowrap; } div.search input { font-size: 93%; width: 11.25em; } div.search input.button { width: auto; margin-left: 5px; } div.search fieldset { border: none; margin: 0; padding: 0; } #topnav div.search fieldset legend { display: none; } /* layout */ #contents { min-width: 37em; } #slashboxes #contents ul { padding-left: 1.25em; margin-bottom: 20px; } body #index #articles { margin: 0 15.6em 1em .8em; position: relative; } /* Header */ #logo h1 a { width: 642px; height: 55px; outline: none; text-decoration: none; background: url("/images/logo-bp-orange4.png") no-repeat left top; } #slogan h2 { color: #555; font-family: sans-serif; font-weight: bold; font-size: 90%; margin: -1em 0 1em 0; } #slashboxes #contents ul { padding-left: 1.25em; margin-bottom: 20px; } /* links */ #links { background: #fff; } #links div.block { background: #fff url("/images/cbl.gif") bottom left no-repeat; } #links div.title { background: #f60 url("/images/ctl_grey.png") top left no-repeat; } #links h4 { font-weight: bold; font-size: 84%; font-family: sans-serif; } #links div.content { padding: .3em .3em .6em .3em; font-size: 85%; line-height: 140%; position: relative; border: 1px #ddd solid; } /* Slashboxes (right sidebar) */ div#slashboxes { float: right; width: 14.5em; margin-left: 1.25em; } div#slashboxes div.block { margin-bottom: 1.25em; font-size: 88%; } div#slashboxes div.block div.title h4 { background: #777 url(//images.slashdot.org/block-title-right.png) no-repeat right top; padding: .1em .1em .1em .8em; color: #fff; font-size: 100%; } div#slashboxes div.block div.title h4 a { color: #fff; } div#slashboxes div.block div.title h4 a:visited { color: #fff; } div#slashboxes div.block div.content { padding: .4em; background: #fff url(//images.slashdot.org/slashbox-bottom-left.png) no-repeat left bottom; } div#slashboxes div.block div.content p { margin: 0;} div#slashboxes div.block div.content ul { margin: 0; padding: 0; list-style-image: none; } div#slashboxes div.block div.content ul li { padding: .1em .1em .3em .5em; /* list-style-image: url(/images//bullet.gif); */ list-style: none; border-top: 1px solid #ddd; } div#slashboxes div.block div.right { padding: .5em .8em .6em .8em; } /* Links (left sidebar) */ div#links { float: left; width: 7.1em; padding-bottom: 10px; margin-right: 0; } div#links div.block div.content ul li a { display: block; padding: .3em .2em .3em .2em; font-size: 100%; border-top: 1px solid #ddd; } div#links div.block div.content ul li a:hover {background: #ef8218 url(//images.slashdot.org/link-arrow.gif) no-repeat right center; color: #fff;} div#links div.block div.content, div#slashboxes div.block div.content { background: #eee; } div#links div.block div.title, div#slashboxes div.block div.title, div#links div.block { /* background: #ef8218 url(/images/article-title-orange-bg.png) repeat-x left top; */ background: #777; } div#links div.block div#sitebox-title, div#links div.block div#navigation-title { background: url(//images.slashdot.org/block-title-right.png) no-repeat right top; } div#links div.block div.content ul { border-bottom: 1px solid #fff; } div#links div.block div.content ul li a { border-top: 1px solid #fff;} /* General */ .generaltitle { background: url("/images/article-title-orange-bg.png") #ef8218 xrepeat-x; } .generaltitle div.title { background: #ef8218; } .generaltitle h3 { color: #fff; font-family: sans-serif; font-size: 120%; /* 16px */ font-weight: bold; } .generaltitle h3 a, .generaltitle h3 a:visited { color: #fff; font-family: sans-serif; font-size: 100%; font-weight: bold; text-decoration: none; } .generalbody { background: #fff; border: 1px solid #eee; border-top: none; } /* articles */ .article div.title { background: #ef8218 url(/images/article-title-orange-bg.png) repeat-x left top; } #journal div.title { background: #ef8218 url(/images/article-title-orange-bg.png) repeat-x left top; } .article div.title h3 { background: url(/images/article-title-left-orange.png) no-repeat left top; color: #fff; font-family: sans-serif; font-size: 120%; } /* titulo secciones */ .article div.title h3 a, .article div.title h3 a:visited { color: #006699; text-decoration: underline; } #journal .article h3 { background: url(//article-title-left-orange.png) no-repeat left top; color: #fff; font-family: sans-serif; font-size: 100%; } .article .details { font-size: 84%; padding: .2em .7em .5em .7em; font-weight: normal; font-family: sans-serif; border: 1px solid #eee; line-height: 130%; /* background: #ddd url("/images/bg_details.gif") repeat-x left top; */ background: #ddd url(//images.slashdot.org/article-details-bg.png) repeat-x left top; } .article .body { font-size: 100%; /* border: 1px solid #eee; */ border-top: none; background: #fff url("/images/bg_white.gif") repeat-x; } div.article div.intro em { display: block; padding: 0 0 0 .85em; margin: .25em 0 .6em 0; font-style: normal; border-left: 3px solid #ddd; } div.article div.intro i a { font-weight: normal; } div.storylinks { margin: 0 0 1.5em 0;} div.storylinks * { line-height: 110%; } div.storylinks div { margin: 0; padding: 0; background: url(//images.barrapunto.com/storylinks-bg.png) repeat-x left bottom; } div.storylinks div ul { margin: 0; padding: .75em 12em .75em .6em; background: url(//images.barrapunto.com/storylinks-right.png) no-repeat right bottom; } div.storylinks ul li { display: inline; padding: 0; list-style: none; background: none; border: none; } div.storylinks ul li.comments { width: 11em; right: 0; position: absolute; margin: -.1em 0; padding: .1em 1em .2em 1em; text-align: right; text-shadow: #000 0 0 0; background: none; } div.storylinks ul li:before {content:"|"; color: #ccc;} div.storylinks ul li.more:before, div.storylinks ul li.comments:before {content:" ";} /* Footer */ #footer { font-family: sans-serif; margin-top: 1.25em; background: #e6e6e6 url(//images.slashdot.org/sections-bg.png) repeat-x left bottom; clear: both; } #footer em { font-size: 85%; float: right; width: 69%; } #footer .search { float: left; width: 15.25em; padding: 1.35em 0 0 1.25em; white-space: nowrap; } #footer legend, #footer fieldset, #footer label { margin: 0; padding: 0; } .checkbox {padding: 0 0 0 0; float: left; } #footer .search input { margin: 0; } #footer .copyright { padding: .85em 1.25em; font-size: x-small; text-align: center; color: #000; } #footer .admin li { border-left: 1px solid #eee; margin: 0; padding: 0; display: inline; list-style: none; } /* Poll */ .barColor { background: #f60; } /* Bottom Nav */ .btmnav { color: #eee; font-size: 85%; } .btmnav ul li a { color: #006699; } .btmnav ul li a:visited { color: #6a6a6a; } /* Journal */ .journaldate { position: relative; } .generalbody #journalgeneric .article { height: auto; margin: 0 0 1em 0; } #journalgeneric div.intro em {display: inline; font-style: italic; border: none; margin: 0; padding: 0;} #journalgeneric div.intro em a {font-style: italic;} #journalgeneric div.storylinks ul li { margin: 0; padding: 0; border: none; } #journalgeneric div.storylinks ul li a { padding: 0 .4em 0 1em; border-left: 1px solid #000; } #journalgeneric div.storylinks ul li.edit a, #journalgeneric div.storylinks ul li.discussion a, #journalgeneric div.storylinks ul li.journalid a { border: none; } #journalslashdot .journaldate {font-weight: bold;} #journalslashdot .title { margin: 0 0 0 .6em; padding: 0; float: left;} #journalslashdot h3 { font-size: 65%; margin: 0; padding: 0; font-family: geneva,verdana,sans-serif;} #journalslashdot .details { float: left; font-style: italic; font-size: 65%; font-family: geneva,verdana,sans-serif; background: transparent; } #journalslashdot .intro { padding: 1em 0 2em 3.7em;} #journalslashdot div.storylinks { margin: 0; padding: 0; background: transparent;font-size: 65%;} #journalslashdot div.storylinks ul { margin: 0; padding: 0; background: transparent;} #journalslashdot div.storylinks ul li { margin: 0; padding: 0; border: none; font-family: geneva,verdana,sans-serif; } #journalslashdot div.storylinks ul li a { padding: 0 .4em 0 1em; border-left: 1px solid #000; } #journalslashdot div.storylinks ul li.edit a, #journalslashdot div.storylinks ul li.discussion a, #journalslashdot div.storylinks ul li.journalid a { border: none; } #journalslashdot .journalpage {font-size: 65.5%; font-family: geneva,verdana,sans-serif; text-align: right;} #journalgrey .journaldate { font-weight: bold; background: #eee; } #journalgrey .details { float: left; font-weight: bold; background: #eee; padding: 0 0.6em 0 0; } #journalgrey .title { background: #eee; padding: 0 0 0 0.6em; } #journalgrey .title h3 { background: #eee; } #journalgrey { border: 3px solid #999; padding: 3px 0 1em;} #journalgrey div.storylinks, #journalblue div.storylinks { background: transparent; } #journalgrey div.storylinks ul, #journalblue div.storylinks ul { background: transparent; } #journalgrey .body, #journalblue .body {min-height: 60px; padding: 0 .5em; } #journalblue div.storylinks ul li, #journalgrey div.storylinks ul li { border: none; margin: 0; padding: 0; } #journalblue .journaldate { background: #369; font-weight: bold; padding: 3px; } #journalblue { border: 3px solid #000; } #journalblue .details { float: left; margin: 0 0.6em 0 0.3em; font-weight: bold; } #journalblue .title h3 { margin: 0; padding: 0; font-family: serif; } .generalbody ul.menu { padding: .5em 0; overflow: auto; } #usermenu ul.menu { padding: .5em 0; overflow: auto; } /* Forms */ #journal input.button, #admin input.button { width: auto; } /* Submit */ #submit .message { margin: 0 0 1em 0; padding: 0; } /* Related */ div.briefarticle {position: relative;} div.briefarticle a span {text-indent: -5000px; position: relative; float: left;} div.briefarticle a span.expanded {width: 15px; background: url(//images.barrapunto.com/login-arrow-expanded.gif) no-repeat 0 50%; } div.briefarticle a span, div.briefarticle a span.condensed {width: 15px; background: url(//images.barrapunto.com/login-arrow-collapsed.gif) no-repeat 0 50%; } div div.briefcomment { padding-left: 4em; padding-top: 0px } /* User section menu */ div#user-section { margin-bottom: 1.25em; white-space: nowrap; font-size: 90%; background: #ef8218 url(/images/userbar-title-orange-bg.png) repeat-x left bottom; } div#user-section * { line-height: 100%; } div#user-section div.content { overflow: hidden; padding: 0; } div#user-section ul { float: left; } div#user-section ul { padding: .3em 0 .1em .6em; margin: 0; } div#user-section ul li { display: inline; list-style: none; } div#user-section ul li.begin {font-weight: bold; padding-left: 0; } div#user-section ul li.name a { font-weight: bold; } div#user-section ul li.name { padding-left: 0; } div#user-section ul li a { font-size: 100%; text-decoration: none; padding: 0 .3em; color: #fff; } div#user-section ul li a:visited { color: #fff; } div#user-section ul li:before {content:"|"; color: #ddd;} div#user-section ul li:first-child:before {content:" ";} div#user-section ul li a#loginlink { padding-left: 1.2em; background: url(//images.slashdot.org/login-arrow-collapsed.gif) no-repeat 0 50%; } div#user-section ul#user-utils { float: right; padding: .3em .6em .3em 0; } div#user-section ul li a#loginlink.expanded { background-image: url(//images.slashdot.org/login-arrow-expanded.gif); } div#user-section div#loginform { clear: both; overflow: hidden; height: 0; width: 100%; padding: 0; margin: 0; } /* Curse the phantom */ @media all and (min-width:0px) { div#user-section div#loginform { height: auto; } head~body div#user-section div#loginform { height: 0; } } div#user-section div#loginform p {display: none;} div#user-section div#loginform form { padding: .75em 0 .25em 0; } div#user-section div#loginform form input { width: 12em; margin-right: 1em;} div#user-section div#loginform form input.button { width: auto; margin-right: 0; } div#user-section div#loginform form label.checkbox { margin-right: 1em; } div#user-section div#loginform form label {display: inline;} div#user-section div#loginform form label.checkbox input { width: auto; margin-right: .25em; } /* System messages */ .indexhead {padding: 0; border-bottom: 1px solid #888; background: #ddd url("//images.barrapunto.com/article-details-bg.png") bottom repeat-x; margin: 0 15.6em 1em .8em; position: relative; } .indexhead div.content { padding: 0; margin: 0; background: url(/images/block-title-left-messages.png) no-repeat left top; } .indexhead div.content p {font-size: 85%; margin: 0; padding: .3em 1.5em; color: #000;} .indexhead div.content a {color: #036;} .indexhead div.content a:visited {color: #6a6a6a;} /* Ads */ .ad1 { width: 750px; background-color: #eee; } .fad1 { width: 468px; margin: 0 240px 0 15px; } .fad2 { float: right; width: 240px; margin: -65px 15px 0 5px; } div#ad2 { text-align: center; margin: 1em 0 1em 0; } div#ad3 { text-align: center; } /* Tags */ div.tags { margin: 1em 0 0 0; padding: 0; font-size: 93%; line-height: 100%; background: #666 url(//images.slashdot.org/block-title-bg.png) repeat-x; } div.tags div.tagtitleclosed, div.tags div.tagtitleopen { margin: 0; padding: .5em 0 .5em 20px; position: relative; background: #fff; } div.tags div.tagtitleopen {color: #fff; background: url(//images.barrapunto.com/block-title-right.png) no-repeat top right; } div.tags div.tagtitleopen a {color: #fff;} div.tags div.tagtitleclosed a { text-decoration: none; } div.tags div.tagtitleclosed .tagmenu a { text-decoration: underline; } div.tags div.tagtitleopen span.tagsheader, div.tags div.tagtitleclosed span.tagsheader {text-indent: -5000px; position:absolute; left: 10px; } div.tags div.tagtitleopen span.tagsheader {width: 15px; background: url(//images.barrapunto.com/block-arrow-expanded.gif) no-repeat 0 50%; } div.tags div.tagtitleclosed span.tagsheader {width: 15px; background: url(//images.barrapunto.com/login-arrow-collapsed.gif) no-repeat 0 50%; } div.tags div.tagtitleopen i a, div.tags div.tagtitleclosed i a {background: transparent;} div.tags div.tagbody { background: #ddd url(//images.barrapunto.com/article-details-bg.png) repeat-x; padding: .85em 0 0 .85em; color: #666; border: #ccc 1px solid; border-top: none; } div.tags div.tagbody input { width: 50%; margin: .5em 0; } div.tags div.tagbody input.button { width: 5em } div.tags div.tagshide { display: none } /* Misc */ pre, code { font-size: 93%; } .tb_col { background: #eee;} .secpref_master td { border-bottom: solid 2px #069;} .secpref_nexus_row { color: #aaa; background: #ccc; } #sectionprefs_message { background: #066; font-size: 120%; color: #fff; font-weight: bold; text-align: right; } #sectionprefs_hdr {text-align: right; background: #069; font-size: 120%; color: #fff; } #sectionprefs_hdr a {float: left; color: #fff;} .popup div.title span, .popup div.title span a { display: none} #vendorStory-26-popup {width: 18.5em; border: 2px solid #c5d1a6} #vendorStory-26-contents {background: #DBE8B8} #vendorStory-26-popup div.title h3 { background: #6d745f url(//images.slashdot.org/vendors-slc.gif) no-repeat top left; } #vendorStory-26-popup div.title h3 a {color: #fff} #vendorStory-26-popup div.title span a, #vendorStory-26-popup div.title span { color: #DBE8B8; } #vendorStory-26-contents a { color: #6d745f; } #vendorStory-26-popup div.details {background: #DBE8B8} #vendorStory-26-title {background: #6D745F; padding: .3em} #sponsorlinks span {color: #069; text-decoration: underline} #contents {margin-top: 2em;} #vendorStory-26-popup #art1, #vendorStory-26-popup #art2, #sponsorlinks #slink1, #sponsorlinks #slink1 .comments {margin: 0} .popup iframe { position:absolute; top: -1px; left:-3px; z-index:-1; width:18.5em; height: 100%; border: none; opacity: 0; } #sectionprefs_hdr span a {float: none; color: #fff; font-size: 10px; font-weight: bold; text-decoration: none;} #sectionprefs_hdr span { margin: -.3em 0; padding: 0 4px; height: 11px; width: 11px;} .ps_23, .ps_22, .ps_26 { display: none; } .curstory { border-top: solid 2px #069; } .popup { border: solid 2px #069; } .popup .data { font-size: 100% } .popup-title {text-align: left; background: #069; font-size: 100%; color: #fff; padding-right: 4em; } .popup-title .buttons { position: absolute; right: 0.2em; } .popup-title .buttons span a {float: none; color: #fff; font-size: 10px; font-weight: bold; text-decoration: none;} .popup-title .buttons span { margin: -.3em 0; padding: 0 4px; height: 11px; width: 11px;} .popup-message { background: #069; color: #fff; } .popup-title a, .popup-message a { color: #fff } .popup-contents { background: #ccc; font-size: 80%; padding: 5px; } .popup-message { background: #069; font-size: 100%; color: #fff; font-weight: bold; text-align: right; } #sectionprefs-contents { background: #fff } #subscribe div.generaltitle div.title {margin-bottom: 0;} #st-display table { background: #069; color: #fff; } blockquote, .quote { margin-bottom: .75em; padding-left: .75em; color: #555; border-left: 3px solid #ddd; position: relative; display: block; } blockquote * { font-style: normal; } .comment > .oneline blockquote { border: 0; padding: 0; margin: 0; } cssutils-1.0/sheets/ll.css0000666000175000017500000002227212126054542014337 0ustar hugohugobody { font-family: "Verdana", "Arial", "XHelvetica", "Helvetica", sans-serif; font-size: 12px; background-color: #333; background-image: url(images/Backdrops/Raster.gif); background-attachment: fixed; background-position: top left; margin: 0px; padding: 0px; } body table.body { background-image: url(images/Backdrops/NautilusBlack.jpg); background-attachment: fixed; background-position: top left; background-repeat: no-repeat; } td.header1, td.header2 { background-color: #0063a8; background-image: url(images/Backdrops/NautilusBlue.jpg); background-attachment: fixed; background-position: top left; background-repeat: no-repeat; } td.blank { background-color: #000; } td.header2 { border-left: 1px solid #000; border-right: 1px solid #000; } td.header2 h1 { color: #fff; font-family: "Trebuchet MS", sans-serif; font-weight: normal; font-size: 50px; margin: 0px 0px 5px 30px; padding: 0px; line-height: 50px; } td.header2 h2 { color: #fff; font-family: "Trebuchet MS", sans-serif; font-weight: normal; font-size: 18px; margin: 0px 0px 10px 30px; padding: 0px; line-height: 18px; } td.header3 { width: 30px; } td.crumbs1, td.crumbs2 { font-family: "Verdana", "Arial", "XHelvetica", "Helvetica", sans-serif; font-size: 11px; background-color: #194b6e; background-image: url(images/Backdrops/NautilusDarkBlue.jpg); background-attachment: fixed; background-position: top left; background-repeat: no-repeat; } td.crumbs1 { border-top: 1px solid #000; } td.crumbs2 { padding: 4px 6px 4px 30px; color: #fff; border-top: 1px solid #000; border-left: 1px solid #000; border-right: 1px solid #000; } td.crumbs2 td.alternate { padding-left: 30px; font-size: 9px; } td.crumbs2 a:link { color: #fff; } td.crumbs2 a:visited { color: #ccc; } td.crumbs2 a:hover { color: #ffc; border-bottom: 1px solid #fff; } td.crumbs2 span.here { color: #fff; font-weight: bold; } td.content { border: 1px solid #000; background-image: url(images/Backdrops/NautilusWhite.jpg); background-attachment: fixed; background-position: top left; background-repeat: no-repeat; background-color: #fff; padding: 20px 30px; line-height: 18px; } table, tr, td, th { margin: 0px; border: 0px solid #000; padding: 0px; } a { text-decoration: none; } td.content a { border-bottom: 1px solid #cc8080; } a:link { color: #900; } a:visited { color: #633; } a:hover, a:active { color: #f00; } td.links { font-family: "Verdana", "Arial", "XHelvetica", "Helvetica", sans-serif; border-top: 1px solid #000; padding: 16px 0px 0px 0px; color: #fff; } td.links ul, td.links li { margin: 0px; padding: 0px; } td.links div, td.links a { display: block; margin: 4px 0px 0px 0px; padding: 2px 2px 2px 10px; white-space: nowrap; color: #fff; font-size: 13px; border-top: 1px solid #333; border-bottom: 1px solid #000; background-color: #858585; background-image: url(images/Backdrops/NautilusGrey.jpg); background-attachment: fixed; background-position: top left; background-repeat: no-repeat; } td.links>div, td.links>a { padding: 6px 2px 6px 10px; } td.links div { position: relative; left: 1px; background-color: #fff; background-image: none; color: #000; } td.links li li div, td.links li li a { background-color: transparent; background-image: none; color: #fff; font-size: 11px; border: 0px; margin: 1px 0px 0px 0px; padding: 0px 3px 2px 3px; } td.links li li div { position: relative; left: 1px; border-bottom: 2px solid #fff; padding: 0px 4px 0px 2px; } td.links ul ul { padding: 0px 0px 0px 30px; } td.links ul ul ul { padding: 0px 0px 0px 20px; } td.links div, td.links a.path { font-weight: bold; } td.links a span.visited { display: none; } td.links a:visited span.visited { display: block; float: right; font-size: 16px; font-weight: normal; padding: 1px 4px 1px 20px; color: #bbb; } td.links a:link { color: #fff; } td.links a:visited { color: #fff; } td.links a:hover { background-color: #0063a8; border-top: 1px solid #333; border-bottom: 1px solid #333; color: #000; background-color: #ccc; background-image: none; } td.links a:hover span.visited { color: #666; } td.links li li a:hover { padding: 0px 2px 0px 2px; } td.links a:active { color: #ff0; } span.tab { color: #ccc; } abbr[title], acronym[title] { border-bottom: 1px solid #ddd; } p, li, dd { font-size: 12px; line-height: 20px; margin: 0px 0px 10px 0px; } dt { font-size: 12px; line-height: 20px; } h1, h2, h3, h4, h5, h6 { color: #0063a8; font-family: "Trebuchet MS", sans-serif; font-weight: normal; } h1 abbr[title], h2 abbr[title], h3 abbr[title], h4 abbr[title], h5 abbr[title], h6 abbr[title], h1 acronym[title], h2 acronym[title], h3 acronym[title], h4 acronym[title], h5 acronym[title], h6 acronym[title] { border-bottom: 0px; } h1 { font-size: 20px; margin: 36px 0 3px 0; line-height: 30px; } h2 { font-size: 18px; margin: 24px 0 2px 0; line-height: 26px; } h3 { font-size: 18px; margin: 16px 0 1px 0; line-height: 22px; } h4, h5, h6 { font-size: 14px; margin: 14px 0 1px 0; line-height: 20px; } td.content div:first-child h1 { margin-top: 0px; } div.section.level1 { } div.section.level1 h1 { margin-right: -30px; border-bottom: 1px solid #666; padding-right: 30px; } ol, ul, dl { margin-top: 10px; margin-bottom: 10px; } li { margin-top: 0px; margin-bottom: 0px; } ul li { list-style-type: square; } ol li { list-style-type: decimal; } dd { margin-left: 20px; margin-top: 0px; margin-bottom: 10px; } em { font-weight: bold; font-style: normal; } address { font-style: normal; font-size: 13px; line-height: 16px; } div.display { margin-left: 1.5em; margin-top: 2px; margin-bottom: 2px; } table.downloads span.note { font-weight: normal; font-size: 9px; } table.downloads { margin-left: 30px; margin-right: 30px; margin-top: 4px; margin-bottom: 10px; font-size: 11px; line-height: 16px; border-bottom: 1px solid #ccc; } table.downloads th, table.downloads td { font-size: 11px; } table.downloads th { padding: 8px 0px 2px 0px; border-bottom: 1px solid #999; } table.downloads th.version { padding: 10px 10px 2px 3px; text-align: left; } table.downloads th.type { color: #666; font-weight: normal; padding: 10px 10px 2px 0px; text-align: left; } table.downloads th.size { color: #666; font-weight: normal; text-align: right; padding: 10px 3px 2px 0px; } table.downloads tr.download { background-color: #fafafa; } table.downloads td.file { padding: 2px 10px 2px 20px; } table.downloads td.type { padding: 2px 10px 2px 0px; } table.downloads td.size { text-align: right; padding: 2px 3px 2px 0px; } pre.prog, pre.tty { margin: 0px 30px 10px 30px; font-size: 12px; line-height: 18px; border: 1px solid #eee; background-color: #fafafa; padding: 1px 5px 2px 5px; overflow: auto; } div.example-title { font-family: "Verdana", "Arial", "XHelvetica", "Helvetica", sans-serif; font-size: 9px; margin-top: -8px; margin-bottom: 4px; margin-right: 30px; color: #666; text-align: right; padding-top: 0px; padding-right: 2px; } span.secnum { background-color: #ccc; color: #fff; font-size: 6px; padding: 1px; vertical-align: middle; } div.class>h1, div.class>h2, div.class>h3, div.class>h4, div.class>h5, div.class>h6, div.function>h1, div.function>h2, div.function>h3, div.function>h4, div.function>h5, div.function>h6, div.method>h1, div.method>h2, div.method>h3, div.method>h4, div.method>h5, div.method>h6, div.property>h6, div.property>h1, div.property>h2, div.property>h3, div.property>h4, div.property>h5 { font-family: monospace; margin-bottom: 0px; padding: 0px 0px 2px 30px; text-indent: -30px; margin-right: -30px; border-bottom: 1px solid #999; padding-right: 30px; } div.method>div.content, div.function>div.content, div.property>div.content, div.class>div.content { margin-left: 30px; } div.class>div.content { border-right: 30px solid #eee; margin-right: -30px; padding-right: 30px; } div.class div.class>div.content { margin-right: 0px; border-right: 0px; padding-right: 0px; } h1 code, h2 code, h3 code, h4 code, h5 code, h6 code { letter-spacing: -1px; } div.class.nodoc, div.method.private, div.method.nodoc, div.function.nodoc, div.property.nodoc { display: none; } code { font-family: "Courier New", monospace; } code.option { } code.literal { } code.function { } code.method { } code.property { } code.class { } var.rep { } code.markup { } code.parameter { } code.module { } code.arg { } code.filename { } code.dirname { } code.hostname { } code.username { } code.prompt { } code.input { } pre code.input { color: #060; } span.application { } cssutils-1.0/sheets/1inherit-ascii.css0000666000175000017500000000004012126054542016526 0ustar hugohugo/* inherited encoding ascii */ cssutils-1.0/sheets/simple.css0000666000175000017500000000046112126054542015215 0ustar hugohugo @import url("fineprint.css") print; @import url(bogus.css); @import url( bogus.css ); @media screen { @three-dee { @background-lighting { azimuth: 30deg; elevation: 190deg; } H1 { color: red } } h1 { foo: bar } h2 { thing: 1; whatsit: 2px } h3 { foo: 2 ! important } } h9 { } b1 & b2 { } cssutils-1.0/sheets/basic.css0000666000175000017500000000014212126054542015001 0ustar hugohugo{ foo: 1.5; bogus: 3, 2, 1; bar-color: #0FEED0; background: #abc; foreground: rgb( 10, 20, 30 ) } cssutils-1.0/sheets/sample_5.css0000666000175000017500000004311612126054542015435 0ustar hugohugo*{border:0} *,#columns,#rss ul,#rss li,#video-frame,.video-right .mootabs_title,.post,.box,p.wp-caption-text,.widget ul li,table#wp-calendar,.wp-caption img,.wp-caption p.wp-caption-text{margin:0} *,select,#nav1 ul,#nav1,#nav1,#nav2 ul,#nav2,#nav2,#search .btn,#header,.video-right .mootabs_title,.box,.post-alt h2,ol.commentlist,table#wp-calendar,.wp-caption img{padding:0} body{font:12px arial,helvetica,sans-serif /18px;background:#333;background:#333 url(http://www.ng4a.com/wp-content/uploads/2010/09/9090981.png) repeat-y scroll center top} a,#search input,#header h1 a,.video-right h2,.video-right .mootabs_title li,.post h3 a,.post-alt h2,.post-alt h3,.post-alt h2 a,.post-alt h3 a,h3.posted a,.comments a,.navigation a,#archivebox h2,#archivebox a,.archivefeed a,.arclist h2,.singletags a,.entry blockquote,p.wp-caption-text,h2.commh2,ul.wooTabs li,.widget ul li a,table#wp-calendar a{color:#000} a,#nav1 li a:hover,#nav2 li a:hover,.video-right .mootabs_title li,ul.wooTabs li a:hover,ul.wooTabs li a.selected{text-decoration:none} a:hover,.singletags a{text-decoration:underline} textarea,select,input{font:12px tahoma,arial,helvetica,sans-serif;border:#ddd 1px solid} textarea,select,input,#rss li,#video-frame,.box .post img,.postedbox,#tabs,#advert_125x125 img,.widget h3,.th{padding:5px} li,#nav1 ul,#nav1,#nav2 ul,#nav2,.video-right .mootabs_title,ol.commentlist{list-style:none} #page,.col1,.col2,.subcol,#nav,#nav-right,#nav2,#video-frame,.post,.box .post p,h3.posted a,.navigation,#archivebox,.arclist ul li,.entry ol li,ol.commentlist li,ol.commentlist .comment-meta,ol.commentlist li,.more_entries,#tabs,.inside,ul.wooTabs,ul.wooTabs li a:hover,ul.wooTabs li a.selected,.inside li,.inside li img.thumbnail,.inside li img.avatar,.fix,.hl-full,.ar,.th,table#wp-calendar a{font:11px tahoma,arial,helvetica,sans-serif} #page,#columns,#nav,#header,.video-left,.video-right,.box .post,.box .post img,.author_photo,.entry img,.inside,.inside li img.thumbnail,.inside li img.avatar,.th{background:#fff} #page{width:939pxpx;margin:4px 27px} #columns{padding:15px 15px 25px} #columns,.video-left h2,.post h3 a,.post-alt h2 a,.post-alt h3 a{font-family:Arial,Helvetica,sans-serif} .col1,.col2,#search,#topbanner,#topbanner img,#rss img,.video-right,.author_photo,.alignright,#sub img,.flickr_badge_image img,.fr,.th{float:left} .col1,.navigation,comment,#slider-holder,#slider-holder .slide,.slider-shelf,.shelf-content{width:595px} .col1,#nav1 a,#nav2 a{padding-right:15px} .col2{width:320px} .subcol{width:150px} .subcol h2,#rss h2{font-size:14px} .subcol h2{padding:0 0 10px} .subcol h2,#nav1 li a:hover,#slider-holder{color:#fff} #nav,.postedbox{height:32px} #nav{padding:2px 0 !important;margin:7px 15px 0 !important;border-bottom:1px solid #ddd;width:930px !important} #nav,#nav2,.postedbox{border-top:1px solid #ddd} #nav-left,#nav-right,#nav1 ul,#nav1,#nav1,#nav1 li,#nav2 ul,#nav2,#nav2,#nav2 li,#search input,#header h1,#rss ul,#video-frame,.video-left,.author_info,.navigation,.alignleft,comment,ol.commentlist li .avatar,.more_entries .wp-pagenavi,.more_entries .wp-pagenavi span.pages,.more_entries .wp-pagenavi .on,.more_entries .wp-pagenavi .current,.more_entries .wp-pagenavi a:visited,.more_entries .wp-pagenavi a:link,.more_entries .wp-pagenavi a,.more_entries .wp-pagenavi .extend,#tabs,ul.wooTabs li,ul.wooTabs li a,.inside li img.thumbnail,.inside li img.avatar,.hl-full,.blog,.fl,.gravatar{float:right} #nav-left,#nav-right,#header h1,.video-left,.video-right,.alignleft,.alignright,ul.wooTabs li{display:inline} #nav-left{width:600px} #nav-right{width:330px} #nav-right,.video-left p,.video-right h2{margin:0 !important} #nav-right,#search,.video-left p,.hl-full{padding:0 !important} #nav1,#nav1 li,#nav2,#search input,span.clicker,.wp-caption p.wp-caption-text{font-size:11px} #nav1 a,#nav2 a,#header h1 a,.video-right .mootabs_title li,img.centered,#tabs,ul.wooTabs li a,#slider-holder .slide-1,#slider-holder .slide a,#slider-holder .slide img,#slider-holder .slider-right,#slider-holder .slider-left,span.clicker,div.aligncenter,.aligncenter{display:block} #nav1 a,#nav2 a,.continue{padding-left:15px} #nav1 li,#nav2 li{line-height:30px} #nav1 li,a.more,.navigation a,ol.commentlist cite{font-weight:bold} #nav1 li,table#wp-calendar caption,table#wp-calendar td#prev,table#wp-calendar td#next{text-transform:uppercase} #nav1 li a:hover,#nav2 li a:hover,.more_entries .wp-pagenavi .extend{background:none} #nav1 li ul,#nav2 li ul,#slider-holder .slide,#slider-holder .slide img,#slider-holder img.full-mask,#slider-holder .slide-content,#slider-holder .slider-right,#slider-holder .slider-left,.slider-shelf,span.clicker,.shelf-content{position:absolute} #nav1 li ul,#nav1 li ul li,#nav2 li ul,#nav2 li ul li{width:15em} #nav1 li ul,#nav1 li.sfhover ul ul,#nav1 li:hover ul ul,#nav1 li.sfhover ul ul ul,#nav1 li.sfhover ul ul,#nav1 li:hover ul ul ul,#nav1 li:hover ul ul,#nav2 li ul,#nav2 li.sfhover ul ul,#nav2 li:hover ul ul,#nav2 li.sfhover ul ul ul,#nav2 li.sfhover ul ul,#nav2 li:hover ul ul ul,#nav2 li:hover ul ul{right:-999em} #nav1 li ul{padding-top:1px} #nav1 li.sfhover ul,#nav1 li:hover ul,#nav1 li li.sfhover ul,#nav1 li.sfhover ul,#nav1 li li:hover ul,#nav1 li:hover ul,#nav1 li li li.sfhover ul,#nav1 li li.sfhover ul,#nav1 li.sfhover ul,#nav1 li li li:hover ul,#nav1 li li:hover ul,#nav1 li:hover ul,#nav1 li.sfhover ul,#nav1 li:hover ul,#nav2 li.sfhover ul,#nav2 li:hover ul,#nav2 li li.sfhover ul,#nav2 li.sfhover ul,#nav2 li li:hover ul,#nav2 li:hover ul,#nav2 li li li.sfhover ul,#nav2 li li.sfhover ul,#nav2 li.sfhover ul,#nav2 li li li:hover ul,#nav2 li li:hover ul,#nav2 li:hover ul{right:auto} #nav1 li ul li,#nav2 li ul li,#search input,ol.commentlist li .even,ul.wooTabs li{background:#fff} #nav1 li ul li,#nav2 li ul li{border-bottom:1px solid #dedbd1;border-left:1px solid #dedbd1;border-right:1px solid #dedbd1;line-height:28px} #nav1 li.hover,#nav1 li:hover,#nav2 li.hover,#nav2 li:hover{position:static} #nav1 li ul ul,#nav2 li ul ul{margin:-29px 0 0 15em} #nav1 li ul ul{border-top:1px solid #dedbd1} #nav1 li.sfhover ul,#nav1 li:hover ul,#nav2 li.sfhover ul,#nav2 li:hover ul{z-index:1000} #nav2{margin:0 0 15px 15px !important;margin:0 0 7px 7px} #nav2,.full{width:930px} #nav2 a{border-left:1px solid #ddd} #nav2 li ul{z-index:99999} #search{height:27px;width:310px;margin:3px 0 0 !important} * html #search{margin-left:8px} #search input,#video-frame,.video-right .mootabs_title li,.box .post,.box .post img,.entry blockquote,.entry img,.more_entries .wp-pagenavi span.pages,.more_entries .wp-pagenavi .on,.more_entries .wp-pagenavi .current,.more_entries .wp-pagenavi a:visited,.more_entries .wp-pagenavi a:link,.more_entries .wp-pagenavi a,#tabs,ul.wooTabs li,.inside li img.thumbnail,.inside li img.avatar,#mpu_banner,#advert_125x125 img,.th,.wp-caption{border:1px solid #ddd} #search input{width:223px;margin:1px 0 0;padding:3px 4px 4px} #search .btn{height:22px;margin:1px 0 0 5px !important;width:69px} #search .btn,span.clicker{width:auto} #search .btn,.more_entries .wp-pagenavi .extend{border:none} #header{height:100px;margin:0 15px} #header h1{line-height:100px;font-size:24px} #header h1 a{height:93px;width:389px;font-size:28px} #topbanner{width:468px;padding:20px 0 0} #rss a,.video-right p,.postedbox,.more_entries .wp-pagenavi span.pages,.more_entries .wp-pagenavi .on,.more_entries .wp-pagenavi .current,.more_entries .wp-pagenavi a:visited,.more_entries .wp-pagenavi a:link,.more_entries .wp-pagenavi a,#footer{color:#666} #rss h2,#rss ul,.video-right,.video-right .mootabs_title li,.author_info,#archivebox,.entry blockquote,#tagcloud{padding:10px} #rss h2,.author_info h3,ol.commentlist p,#respond p{margin-bottom:10px} #rss li,.widget ul{border:none} #rss img{padding-right:10px} #rss img,.singletags{margin-top:5px} #video-frame,.more_entries .wp-pagenavi span.pages,.more_entries .wp-pagenavi .on,.more_entries .wp-pagenavi .current,.more_entries .wp-pagenavi a:visited,.more_entries .wp-pagenavi a:link,.more_entries .wp-pagenavi a{background:#eee} #video-frame{width:583px;height:auto} .video-left{width:285px !important} .video-left h2,#slider-holder img.full-mask{display:none} .video-right{width:273px;min-height:214px} .video-right h2{font:11px tahoma,arial,helvetica,sans-serif bold !important /normal} .video-right h2,.post h2,h2.commh2{padding:0 0 5px} .video-right p{font:11px tahoma,arial,helvetica,sans-serif /12px;padding:0 0 20px;margin:10px 0 0} .video-right h3{font:18px arial,helvetica,sans-serif normal !important bold /20px} .video-right h3,.archivefeed{padding-bottom:0} .video-right .mootabs_title{border-top:1px solid #FFF} .video-right .mootabs_title li{font:11px tahoma,arial,helvetica,sans-serif normal /10px;margin:0 0 5px} .video-right .mootabs_title li,ol.commentlist li.alt{background:#f7f7f7} .video-right .mootabs_title li,ul.wooTabs li,#slider-holder .slider-right,#slider-holder .slider-left,span.clicker,.shelf-content img{cursor:pointer} .video-right .mootabs_title li.active,.video-right .mootabs_title li:hover{background:#ddd url(images/ico-watch.gif) left center no-repeat !important} .post h2{font:18px arial,helvetica,sans-serif} .post h3{font:14px arial,helvetica,sans-serif} .post p,.post-alt p,.arclist h2{padding:0 0 15px} .post p{line-height:15px} .box,.postedbox,.more_entries .wp-pagenavi,#footer,.fix,.hl-full{clear:both} .box{height:1%} .box .post{width:288px} .box .post h2 a,#archivebox h2,#archivebox h3,#archivebox h4{color:#333} .box .post h2{font-size:16px} .box .post h2,.inside li img.thumbnail,.inside li img.avatar{padding:3px} .box .post h2,ol.commentlist p{line-height:20px} .box .post p{padding:5px 10px} .box .post img,.wp-caption{margin:10px} .box .post img{float:none} .post-alt,h2.commh2{margin:0 0 15px 0} .post-alt h2{font-size:22px;line-height:26px} .post-alt h3{font:15px arial,helvetica,sans-serif /18px} .post-alt h3,.post-alt h4{padding-bottom:5px} .post-alt h4{font:14px arial,helvetica,sans-serif /16px} .post-alt p.post_date,.comments a{font:11px tahoma,arial,helvetica,sans-serif normal} .post-alt p.post_date,table#wp-calendar td{color:#999} .post-alt p.post_date{padding:5px 0 10px} .postedbox{font:11px tahoma,arial,helvetica,sans-serif normal normal /15px} .postedbox,.author_info,#tabs,#mpu_banner{background:#eee} .postedbox,.archivefeed{margin-top:10px} .postedbox,.entry p,.ar{text-align:right} .box .post h3.posted{border-bottom:none} .author_info{width:573px;border:1px solid #ccc} .author_info,#archivebox,.arclist ul,.wp-caption img,.entry ol,.entry ul,comment,#mpu_banner,.widget,.between{margin-bottom:15px} .author_photo{margin:0 10px 0 0} .author_photo img{border:1px solid #666} .continue,.archivefeed a{font:11px tahoma,arial,helvetica,sans-serif bold} .continue{background:url(images/ico-arrow.gif) no-repeat left} .comments{padding-bottom:0 !important} .comments a{background:url("images/ico-comm.gif") no-repeat scroll left center transparent;padding:0 0 0 18px} .comments a{margin-right:5px} a.more{background:url(images/ico-arrow2.gif) no-repeat right;font-size:1.2em;padding:0 0 0 20px} a.more,.arclist ul li{color:#c0c0c0} .post-alt blog{background:url(http://gs-shoping.net/blog/wp-content/uploads/2010/04/item_list_back.gif) repeat-x right top} .navigation{margin-bottom:-30px;padding:15px 0} .navigation a{font-size:1em} .alignleft,.entry ul li{margin-right:15px} .alignright{margin-left:15px} img.wp-smiley{padding:0!important;border:none!important} img.centered,div.aligncenter,.aligncenter{margin-left:auto;margin-right:auto} #archivebox,ul.wooTabs li a{color:#000 !important} #archivebox h3 em,#archivebox h2 em,#archivebox h4 em,table#wp-calendar caption{font-weight:normal} .archivefeed a{background:url(images/ico-arcfeed.gif) center right no-repeat;height:20px;padding:3px 22px 1px 0} .arclist{width:260px;margin:0 0 25px;padding:5px 0 0} .arclist,.inside li,.flickr_badge_image img,.widget ul li,.hl3{border-bottom:1px solid #eee} .arclist ul li{padding:8px 10px 8px 0;border-top:1px solid #eee} .singletags a{font:11px tahoma,arial,helvetica,sans-serif normal !important} .entry p{font:11px tahoma,arial,helvetica,sans-serif /18px} .entry blockquote{font:12px tahoma,arial,helvetica,sans-serif italic;width:85%;margin:5px auto 15px} .entry blockquote,#footer{background:#f7f7f7} .entry blockquote p,.entry ol,.entry ul,ol.commentlist,#respond p{font:11px tahoma,arial,helvetica,sans-serif /20px} .entry blockquote p,.widget ul{padding:5px 0} .entry blockquote p{margin:5px 0} .entry img{padding:4px} p.wp-caption-text{font:11px tahoma,arial,helvetica,sans-serif italic;padding:0 0 15px 0} p.wp-caption-text,.ac,table#wp-calendar th,table#wp-calendar td,.wp-caption{text-align:center} .entry ol li ol,.entry ul li ul{margin-bottom:0;padding-top:3px} .entry ul li{list-style:circle inside !important} .entry ul li,.entry ol li{padding:3px 0} .entry ul li ul li{list-style:disc inside !important} .entry ol li{list-style:decimal inside !important} .entry ol li ol li{list-style:decimal-leading-zero inside !important} h2.commh2{font:1.5em tahoma,arial,helvetica,sans-serif bold !important} ol.commentlist{margin:10px 0 20px} ol.commentlist p{font-size:12px;margin-top:17px} ol.commentlist cite{font-style:normal} ol.commentlist li{padding:20px 10px 10px 10px;margin:10px 0 10px;padding:15px 10px 10px} ol.commentlist li.alt{padding:10px 10px 0 10px !important} ol.commentlist li.alt,ol.commentlist li,.widget select{border:1px solid #eee} ol.commentlist li .avatar,.gravatar{border:1px solid #000} ol.commentlist li .avatar{margin-left:10px} .commentlist li ul li,.gravatar{margin-right:10px} .commentlist li ul li{margin-left:-5px} ol.commentlist li .reply{padding:0 0 5px 0} .more_entries .wp-pagenavi a{text-decoration:none !important} .more_entries .wp-pagenavi a:hover,.more_entries .wp-pagenavi .on,.more_entries .wp-pagenavi .current{background:#9f9f9f !important} .more_entries .wp-pagenavi span.pages,.more_entries .wp-pagenavi .on,.more_entries .wp-pagenavi .current,.more_entries .wp-pagenavi a:visited,.more_entries .wp-pagenavi a:link,.more_entries .wp-pagenavi a{font:11px tahoma,arial,helvetica,sans-serif /13px;padding:4px !important;margin-right:4px} #tabs{width:308px !important;height:auto !important} .inside{width:293px !important;padding:5px !important;border:1px solid #ddd !important;margin:5px 0 0 !important} ul.wooTabs,.hl-full{width:100%} ul.wooTabs{padding:0 0 7px 0} ul.wooTabs li{font:11px tahoma,arial,helvetica,sans-serif bold /34px;margin:0 2px 0 0 !important} ul.wooTabs li a{font:11px tahoma,arial,helvetica,sans-serif /31px;padding:0 6px} ul.wooTabs li a:hover,ul.wooTabs li a.selected{color:#fff !important} #sub img{margin:0 8px 8px 0} .inside li{padding:8px} .inside li img.thumbnail,.inside li img.avatar{margin:0 0 0 8px} #mpu_banner{width:300px;height:250px;padding:9px} #advert_125x125{padding-left:14px} #advert_125x125 img{margin:0 14px 14px 0} #flickr .wrap,.textwidget{padding-top:15px} .widget ul{margin:0 0 10px 0} .widget ul li{padding:5px 5px 5px 0} .widget select{width:148px;margin:15px 0 0 0} #footer{font:11px tahoma,arial,helvetica,sans-serif /35px;height:35px;margin:0 15px 15px;padding:0 15px} #footer img{vertical-align:middle} .fix,.hl-full{height:1px} .fix{margin:0 0 -1px} .fix,#slider-holder,#slider-holder .slide{overflow:hidden} .hl{border-bottom:2px solid #eee} .hl-full{margin:0 0 15px} .hl2{border-bottom:3px solid #c0c0c0} .blog{width:100% !important} .between{height:2px} .th{margin:0 10px 5px 0} table#wp-calendar{border-collapse:collapse} table#wp-calendar caption,table#wp-calendar th{font-size:0.91667em;line-height:1.72727em;border-bottom:1px dotted #ddd} table#wp-calendar caption{letter-spacing:2px;background:#e5e5e5} table#wp-calendar th,table#wp-calendar td{width:26px} table#wp-calendar th{background:#f2f2f2} table#wp-calendar td{line-height:1.66667em} table#wp-calendar td#prev,table#wp-calendar td#next{width:78px;font-size:0.83333em;letter-spacing:1px} .jdGallery a{cursor:pointer !important} #slider-holder{position:relative;margin-bottom:20px} #slider-holder,#slider-holder .slide{height:270px} #slider-holder,#slider-holder .slide,#slider-holder .slide-content,.slider-shelf,span.clicker{background:#000} #slider-holder .slide,#slider-holder .slide-1,#slider-holder .slide img,#slider-holder img.full-mask,#slider-holder .slider-left,#slider-holder .slider-right,.slider-shelf,span.clicker,.shelf-content{top:0} #slider-holder .slide{left:595px;z-index:1} #slider-holder .slide-1,#slider-holder .slide img,#slider-holder img.full-mask,#slider-holder .slide-content,#slider-holder .slider-right{right:0} #slider-holder img.full-mask{z-index:0} #slider-holder .slide-content{top:270px;height:700px;width:575px;padding:10px 10px 0 10px} #slider-holder .slide-content,#slider-holder .slider-right,#slider-holder .slider-left,.slider-shelf,span.clicker{z-index:999} #slider-holder .slider-right,#slider-holder .slider-left{height:150px;width:120px;color:white} #slider-holder .slider-left{left:0;background:url('images/fleche1.png') no-repeat center bottom} * html #slider-holder .slider-left{background:url('images/fleche1.gif') no-repeat center left} #slider-holder .slider-right{background:url('images/fleche2.png') no-repeat center bottom} * html #slider-holder .slider-right{background:url('images/fleche2.gif') no-repeat center right} .slider-shelf{overflow:visible} span.clicker{right:120px;padding:2px 10px} .shelf-content{z-index:9999} .shelf-content img{border:1px solid #fff;margin:10px 10px 0 0} .shelf-title{padding:0 10px} .wp-caption{background:#f3f3f3;padding-top:4px;-moz-border-radius:3px;-khtml-border-radius:3px;-webkit-border-radius:3px;border-radius:3px} .wp-caption img{border:0 none} .wp-caption p.wp-caption-text{line-height:17px;padding:0 4px 5px} cssutils-1.0/sheets/single-color.css0000666000175000017500000000016112126054542016316 0ustar hugohugo@media aural { BLOCKQUOTE:after { content: url("beautiful-music.wav") } } a { content: url("other.wav") } cssutils-1.0/sheets/xhtml22.css0000666000175000017500000001133112126054542015222 0ustar hugohugo@namespace "http://www.w3.org/2002/06/xhtml2/"; /* A sample style sheet for XHTML 2.0 This style sheet describes a very incomplete, sample rendering of XHTML 2.0 elements. Editor: Masayasu Ishikawa Revision: $Id$ */ /* new elements */ section, h, nl, label, l, blockcode, separator, di { display: block } section, h, nl, label, l, blockcode, di { unicode-bidi: embed } nl { margin: 1.33em 0 } summary, standby, handler { display: none } blockcode { font-family: monospace; white-space: pre } separator { border-bottom: thin black solid; border: 1px } h { display: block; font-weight: bolder; font-family: sans-serif } h1, h2, h3, h4, h5, h6 { font-family: sans-serif; font-weight: bolder } body h, h1 { font-size: 2em; margin: 0.67em 0 } section h, h2 { font-size: 1.5em; margin: 0.83em 0 } section section h, h3 { font-size: 1.17em; margin: 1em 0 } section section section h, h4, p, blockquote, ul, ol, dl { margin: 1.33em 0 } section section section section h, h5 { font-size: 0.83em; line-height: 1.17em; margin: 1.67em 0 } section section section section section h, h6 { font-size: 0.67em; margin: 2.33em 0 } *[edit="deleted"] { display: none } /* no special presentation by default *[edit="inserted"] { } *[edit="changed"] { } *[edit="moved"] { } */ /* experimental navigation list style */ nl { height: 1.5em; overflow: hidden; margin: 0; line-height: normal !important; white-space: nowrap; text-align: start; cursor: default; border-width: 2px !important; border-style: inset !important; vertical-align: baseline; padding: 0 } nl:hover { height: auto; overflow: visible } nl>li, nl>label { display: block; min-height: 1em; line-height: normal !important } nl>li, nl>label { padding: 0 5px 0 3px } nl>li { margin-left: 1em } nl>label { font-weight: bold } nl>nl>label { display: block; line-height: normal !important; font-style: italic; font-weight: bold } nl>nl>li { padding-left: 2em; font-style: normal; font-weight: normal } /* inherited elements */ html, body, div, p, h1, h2, h3, h4, h5, h6, address, blockquote, pre, ol, ul, dl , dt, dd { display: block } li { display: list-item } head, style, link, meta { display: none } table { display: table; border-spacing: 0; border-top: thin black solid; border-left: thin black solid } tr { display: table-row } thead { display: table-header-group } tbody { display: table-row-group } tfoot { display: table-footer-group } col { display: table-column } colgroup { display: table-column-group } td, th { display: table-cell; border-right: thin black solid; border-bottom: thin black solid } caption { display: table-caption } table:hover summary { display: block } th { font-weight: bolder; text-align: center } caption { text-align: center } body { padding: 8px; line-height: 1.2 } strong { font-weight: bolder } blockquote { margin-left: 4em; margin-right: 4em } cite, em, q, var, address { font-style: italic } pre code, kbd, samp { font-family: monospace } pre { white-space: pre } sub, sup { font-size: smaller } sub { vertical-align: sub } sup { vertical-align: super } ol, ul, dd { margin-left: 4em } ol { list-style-type: decimal } ol ul, ul ol, ul ul, ol ol { margin-top: 0; margin-bottom: 0 } abbr[title] { border-bottom: dotted 1px } :link { text-decoration: underline; color: blue } :focus { outline: thin dotted invert } /* Hover effects should be default */ :link:hover, :link:visited { color: #b7f } /* begin bidirectionality settings (do not change) */ *[dir="ltr"] { direction: ltr; unicode-bidi: embed } *[dir="rtl"] { direction: rtl; unicode-bidi: embed } *[dir="lro"] { direction: ltr; unicode-bidi: bidi-override } *[dir="rlo"] { direction: rtl; unicode-bidi: bidi-override } /* block-level elements */ body, div, p, hr, h1, h2, h3, h4, h5, h6, address, blockquote, pre, ol, ul, li, di, dt, dd, table, thead, tbody, tfoot, tr, td, th, col, colgroup, caption, obje ct, summary, standby, blockcode { unicode-bidi: embed }cssutils-1.0/sheets/yuck.css0000666000175000017500000000016512126054542014700 0ustar hugohugoE[class~="hipster"][thing~="bob"]#myid { att1: hi; foo: bar } E:lang(c)#unique E.hipster#myid { att1: hi; foo: bar } cssutils-1.0/sheets/html.css0000666000175000017500000000017412126054542014671 0ustar hugohugobody { color: red } body { color: blue } body { color: pink } body { color: green } cssutils-1.0/sheets/tigris2.css0000666000175000017500000002157212126054542015315 0ustar hugohugo/* contains rules unsuitable for Netscape 4.x; simpler rules are in ns4_only.css. see */ /* colors, backgrounds, borders, link indication */ body { background: #fff; color: #000; } .app h3, .app h4, .tabs td, .tabs th, .functnbar { background-image: url(../images/nw_min.gif); background-repeat: no-repeat; } #toptabs td, #toptabs th { background-image: url(../images/nw_min_036.gif); } #navcolumn .body div, body.docs #toc li li { background-image: url(../images/strich.gif); background-repeat: no-repeat; background-position: 0.5em 0.5em; } #search .body div, .body .heading { background-image: none; } .app h3, .app h4 { color: #fff; } .app h3, #banner td { background-color: #036; color: #fff; } body #banner td a { color: #fff !important; } .app h4 { background-color: #888; } .a td { background: #ddd; } .b td { background: #efefef; } table, th, td { border: none; } .mtb { border-top: solid 1px #ddd; } div.colbar { background: #bbb; } #banner { border-top: 1px solid #369; } .toolgroup { background: #eee; } .toolgroup .label { border-bottom: 1px solid #666; border-right: 1px solid #666; background: #ccc; } .toolgroup .body { border-right: 1px solid #aaa; border-bottom: 1px solid #aaa; } #mytools .label, #projecttools .label, #admintools .label { background: #fff; border-top: 1px solid #666; border-right: none; border-bottom: none; border-left: 1px solid #666; } #mytools .body, #projecttools .body, #admintools .body { background: #fff; border-top: none; border-right: none; border-bottom: none; border-left: 1px solid #666; } #mytools, #projecttools, #admintools { border-right: 1px solid #aaa; border-bottom: 1px solid #aaa; } #helptext { background: #ffc; } #helptext .label { border-bottom: 1px solid #996; border-right: 1px solid #996; background: #cc9; } #helptext .body { border-bottom: 1px solid #cc9; border-right: 1px solid #cc9; } #breadcrumbs { border-top: 1px solid #fff; background-color: #ccc; } #main { border-top: 1px solid #999; } #rightcol div.www, #rightcol div.help { border: 1px solid #ddd; } body.docs div.docs { background: #fff; border-left: 1px solid #ddd; border-top: 1px solid #ddd; } body.docs { background: #eee url(../images/help_logo.gif) top right no-repeat !important; } .docs h3, .docs h4 { border-top: solid 1px #000; } #alerterrormessage { background: url(../images/icon_alert.gif) top left no-repeat !important; } .functnbar { background-color: #aaa; } .functnbar2, .functnbar3 { background: #aaa url(../images/sw_min.gif) no-repeat bottom left; } .functnbar3 { background-color: #ddd; } .functnbar, .functnbar2, .functnbar3 { color: #000; } .functnbar a, .functnbar2 a, .functnbar3 a { color: #000; text-decoration: underline; } #topmodule { background: #ddd; border-top: 1px solid #fff; border-bottom: 1px solid #aaa; } #topmodule #issueid { border-right: 1px solid #aaa; } a:link, #navcolumn a:visited, .app a:visited, .tasknav a:visited { color: blue; } a:link.selfref, a:visited.selfref { color: #555 !important; text-decoration: none; } a:active, a:hover, #leftcol a:active, #leftcol a:hover { color: #f30 !important; } #login a:link, #login a:visited { color: white; text-decoration: underline; } #banner a:active, #banner a:hover { color: #f90 !important; } #leftcol a, #breadcrumbs a { text-decoration: none; } #apphead h2 em { color: #777; } .app th { background-color: #bbb; } .tabs th { border-right: 1px solid #333; background-color: #ddd; color: #fff; } .tabs td { background-color: #999; border-bottom: 1px solid #fff; border-right: 1px solid #fff; } .tabs { border-bottom: 6px #ddd solid; } .tabs th, .tabs th a:link, .tabs th a:visited { color: #555; } .tabs td, .tabs td a:link, .tabs td a:visited { color: #fff; } .tabs a { text-decoration: none; } #toptabs td { border-bottom: 1px solid #666; border-right: 1px solid #333; border-left: 1px solid #036; } #toptabs th { border-left: 1px solid #036; } .axial th { background-color: #ddd; color: black; } .alert { color: #c00; } .confirm { color: green; } .info { color: blue; } .selection { background: #ffc; } #login { color: #fff; } h4 a:link, h4 a:visited { text-decoration: underline; color: #fff; } /* font and text properties, exclusive of link indication, alignment, text-indent */ body, th, td, input, select, textarea, h2 small { font-family: Verdana, Helvetica, Arial, sans-serif; } code, pre { font-family: 'Andale Mono', Courier, monospace; } html body, body th, body td, textarea, h2 small, .app h3, .app h4, #rightcol h3, #bodycol pre, #bodycol code { font-size: x-small; font-size: small; voice-family: "\"}\""; voice-family: inherit; } html>body, html>body th, html>body td, html>body input, html>body select, html>body textarea, html>body h2 small, html>body .app h3, html>body .app h4, html>body #rightcol h3, html>body #bodycol pre, html>body #bodycol code { font-size: small; } small, div#footer td, div#login, div.tabs th, div.tabs td, input, select, .paginate, .functnbar, .functnbar2, .functnbar3, #breadcrumbs td, .courtesylinks, #rightcol div.help, .colbar, .tasknav, body.docs div#toc, #leftcol { font-size: xx-small; font-size: x-small; voice-family: "\"}\""; voice-family: inherit; } html>body small, html>body div#footer td, html>body div#login, html>body div.tabs th, html>body div.tabs td, html>body input, html>body select, html>body .paginate, html>body .functnbar, html>body .functnbar2, html>body .functnbar3, html>body #breadcrumbs td, html>body .courtesylinks, html>body #rightcol div.help, html>body .colbar, html>body .tasknav, html>body.docs #toc, html>body #leftcol { font-size: x-small; } #bodycol h2 { font-family: Tahoma, Verdana, Helvetica, Arial, sans-serif; font-size: 1.5em; font-weight: normal; } .tabs td, .tabs th, dt, .tasknav .selfref, #login .username, .selection { font-weight: bold; } h4 { font-size: 1em; } #apphead h2 em { font-style: normal; } /* box properties (exclusive of borders), positioning, alignments, list types, text-indent */ #bodycol h2 { margin-top: 0.3em; margin-bottom: 0.5em; } p, ul, ol, dl { margin-top: 0.67em; margin-bottom: 0.67em; } h3, h4 { margin-bottom: 0; } form { margin-top: 0; margin-bottom: 0; } #bodycol { padding-left: 12px; padding-right: 12px; width: 100%; width: auto; voice-family: "\"}\""; voice-family: inherit; } html>body #bodycol { width: auto; } .docs { line-height: 1.4; } .app h3, .app h4 { padding: 5px; margin-right: 2px; margin-left: 2px; } .h3 p, .h4 p, .h3 dt, .h4 dt { margin-right: 7px; margin-left: 7px; } .tasknav { margin-bottom: 1.33em; } div.colbar { padding: 3px; margin: 2px 2px 0; } .tabs { margin-top: 0.67em; margin-right: 2px; margin-left: 2px; } .tabs td, .tabs th { padding: 3px 9px; } #toptabs { margin: 0; padding-top: 0.67em; padding-left: 8px; } #breadcrumbs td { padding: 2px 8px; } #rightcol div.www, #rightcol div.help { padding: 0 0.5em; } body.docs #toc { position: absolute; top: 15px; left: 0px; width: 120px; padding: 0 20px 0 0; } body.docs #toc ul, #toc ol { margin-left: 0; padding-left: 0; } body.docs #toc li { margin-top: 7px; padding-left: 10px; list-style-type: none; } body.docs div.docs { margin: 61px 0 0 150px; padding: 1em 2em 1em 1em !important; } .docs p+p { text-indent: 5%; margin-top: -0.67em; } .docs h3, .docs h4 { margin-bottom: 0.1em; padding-top: 0.3em; } #alerterrormessage { padding-left: 100px; } .functnbar, .functnbar2, .functnbar3 { padding: 5px; margin: 0.67em 2px; } #topmodule td { vertical-align: middle; padding: 2px 8px; } body { padding: 1em; } body.composite, body.docs { margin: 0; padding: 0; } th, td { text-align: left; vertical-align: top; } .right { text-align: right !important; } .center { text-align: center !important; } .axial th { text-align: right; } .app .axial td th { text-align: left; } body td .stb { margin-top: 1em; text-indent: 0; } body td .mtb { margin-top: 2em; text-indent: 0; } .courtesylinks { margin-top: 1em; padding-top: 1em; } dd { margin-bottom: 0.67em; } .toolgroup { margin-bottom: 6px; } .toolgroup .body { padding: 4px 4px 4px 0; } .toolgroup .label { padding: 4px; } .toolgroup .body div { padding-bottom: 0.3em; padding-left: 1em; } #banner td { vertical-align: bottom; } #mytools .body, #projecttools .body, #admintools .body { padding-top: 0; } #mytools, #projecttools, #admintools { margin: -4px 0 6px -4px; padding: 6px; border-right: 1px solid #aaa; border-bottom: 1px solid #aaa; } cssutils-1.0/sheets/1ascii.css0000666000175000017500000000010712126054542015072 0ustar hugohugo@charset "ascii"; @import "1inherit-ascii.css"; @import "1utf.css"; cssutils-1.0/sheets/2inherit-iso.css0000666000175000017500000000004712126054542016240 0ustar hugohugo/* 2 inherited encoding iso-8859-1 */ cssutils-1.0/sheets/import/0000755000175000017500000000000012416502415014516 5ustar hugohugocssutils-1.0/sheets/import/import-impossible.css0000666000175000017500000000013012126054542020705 0ustar hugohugo@namespace "y"; @media tv {} .import4 { background: url(images2/example2.gif); } cssutils-1.0/sheets/import/import2.css0000666000175000017500000000065312126054542016635 0ustar hugohugo@import "../import3.css"; @import "import-impossible.css" print; .import2 { /* sheets/import2.css */ background: url(http://example.com/images/example.gif); background: url(//example.com/images/example.gif); background: url(/images/example.gif); background: url(images2/example.gif); background: url(./images2/example.gif); background: url(../images/example.gif); background: url(./../images/example.gif); }cssutils-1.0/sheets/import/images2/0000755000175000017500000000000012416502415016045 5ustar hugohugocssutils-1.0/sheets/import/images2/example2.gif0000666000175000017500000000000012126054542020244 0ustar hugohugocssutils-1.0/sheets/test-unicode.css0000666000175000017500000002206412126054542016332 0ustar hugohugo@charset "UTF-16"; /* * This is a test file only - it doesn't represent a usable style sheet, * it is meant only to test the lexer and parser. */ @import url(foo.css); @import "bar.css"; @import url("fineprint.css") print; @import url("bluish.css") projection, tv; @import url(http://www.steadystate.com/primary.css) screen, tv; @import "sounds_good.css" aural; E[class~="hipster"][thing~="bob"]#myid { att1: hi; foo: bar; } E:lang(c)#unique E.hipster#myid { att1: hi; foo: bar; } @media tv, radio { @page { foo: bar; } H1 { fillcolor: blue } } /* @import url(bad_import.css); */ H1, H2 {color:green;background-color:blue} H1, H2 {color: green;background-color: blue} H3, H4 & H5 { color: red; a: red; b: red; c: red; d: red; e: red; f: red; g: red; h: red; i: red; j: red; k: red; l: red; m: red; n: red; o: red; } H6 { color: black } H7 & H8 { color: red; @page { foo: bar } } P[example="public class foo\ {\ private int x;\ \ foo(int x) {\ this.x = x;\ }\ \ }"] { color: red } H1 { color: red; font-style: 12pt } /* Invalid value: 12pt */ P { color: blue; font-vendor: any; /* Invalid prop.: font-vendor */ font-variant: small-caps } EM EM { font-style: normal } H1 { color: red; rotation: 70minutes } IMG { float: left } /* correct CSS2 */ IMG { float: left here } /* "here" is not a value of 'float' */ IMG { background: "red" } /* keywords cannot be quoted in CSS2 */ IMG { border-width: 3 } /* a unit must be specified for length values */ @three-dee { @background-lighting { azimuth: 30deg; elevation: 190deg; } H1 { color: red } } H1 { color: blue !important } P {counter-increment: par-num ! important } H1 {counter-reset: par-num} P:before {content: counter(par-num, upper-roman) ". "} EM { color: #f00 } /* #rgb */ EM { color: #ff0000 } /* #rrggbb */ EM { color: rgb(255,0,0) } /* integer range 0 - 255 */ EM { color: rgb(100%, 0%, 0%) } /* float range 0.0% - 100.0% */ FOO { content: "Argos\t\n" } R1 { a: U+aaaaa-ffffff } R2 { a: U+?????? } R3 { a: U+f????? } R4 { a: U+fffff? } R5 { a: U+ffffff } * {} /* a=0 b=0 c=0 -> specificity = 0 */ LI {} /* a=0 b=0 c=1 -> specificity = 1 */ UL LI {} /* a=0 b=0 c=2 -> specificity = 2 */ UL OL+LI {} /* a=0 b=0 c=3 -> specificity = 3 */ H1 + *[REL=up]{} /* a=0 b=1 c=1 -> specificity = 11 */ UL OL LI.red {} /* a=0 b=1 c=3 -> specificity = 13 */ LI.red.level {} /* a=0 b=2 c=1 -> specificity = 21 */ #x34y {} /* a=1 b=0 c=0 -> specificity = 100 */ DIV > P:first-child { text-indent: 0 } HTML:lang(fr) { quotes: ' ' ' ' } HTML:lang(de) { quotes: '' '' '\2039' '\203A' } :lang(fr) > Q { quotes: ' ' ' ' } :lang(de) > Q { quotes: '' '' '\2039' '\203A' } H1 { bogus: alpha & beta } H2 { foo: inherit; ;; } /* * Counters and stuff */ P {counter-increment: par-num} H1 {counter-reset: par-num} P:before {content: counter(par-num, upper-roman) ". "} H1:before { content: "Chapter " counter(chapter) ". "; counter-increment: chapter; /* Add 1 to chapter */ counter-reset: section; /* Set section to 0 */ } H2:before { content: counter(chapter) "." counter(section) " "; counter-increment: section; } OL { counter-reset: item } LI { display: block } LI:before { content: counters(item, "."); counter-increment: item } H1:before { content: counter(chno, upper-latin) ". " } H2:before { content: counter(section, upper-roman) " - " } BLOCKQUOTE:after { content: " [" counter(bq, hebrew) "]" } DIV.note:before { content: counter(notecntr, disc) " " } P:before { content: counter(p, none) } /* * Content stuff (w/attr function) */ @media aural { BLOCKQUOTE:after { content: url("beautiful-music.wav") } } IMG:before { content: attr(alt) } H1:before { display: block; text-align: center; content: "chapter\A hoofdstuk\A chapitre" } /* * Clip (w/rect function) */ P { clip: rect(5px, 10px, 10px, 5px); } P { clip: rect(5px, -5px, 10px, 5px); } P { font-size: 12pt; line-height: 12pt } P:first-letter { font-size: 200%; font-style: italic; font-weight: bold; float: left; bogus: foo(p1, p2, "hello", url(primary.css)) } SPAN { text-transform: uppercase } P { color: red; font-size: 12pt } P:first-letter { color: green; font-size: 200% } P:first-line { color: blue } P { text-indent:-.5in; text-indent:-3.5in }cssutils-1.0/sheets/1.css0000666000175000017500000000007312126054542014063 0ustar hugohugo@charset "utf-8"; /*äöüß*/ /* κουρος */ cssutils-1.0/sheets/1inherit-utf8.css0000666000175000017500000000003612126054542016331 0ustar hugohugo/* inherit encoding utf-8 */ cssutils-1.0/COPYING.LESSER0000666000175000017500000001717212126054542013755 0ustar hugohugo GNU LESSER 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. This version of the GNU Lesser General Public License incorporates the terms and conditions of version 3 of the GNU General Public License, supplemented by the additional permissions listed below. 0. Additional Definitions. As used herein, "this License" refers to version 3 of the GNU Lesser General Public License, and the "GNU GPL" refers to version 3 of the GNU General Public License. "The Library" refers to a covered work governed by this License, other than an Application or a Combined Work as defined below. An "Application" is any work that makes use of an interface provided by the Library, but which is not otherwise based on the Library. Defining a subclass of a class defined by the Library is deemed a mode of using an interface provided by the Library. A "Combined Work" is a work produced by combining or linking an Application with the Library. The particular version of the Library with which the Combined Work was made is also called the "Linked Version". The "Minimal Corresponding Source" for a Combined Work means the Corresponding Source for the Combined Work, excluding any source code for portions of the Combined Work that, considered in isolation, are based on the Application, and not on the Linked Version. The "Corresponding Application Code" for a Combined Work means the object code and/or source code for the Application, including any data and utility programs needed for reproducing the Combined Work from the Application, but excluding the System Libraries of the Combined Work. 1. Exception to Section 3 of the GNU GPL. You may convey a covered work under sections 3 and 4 of this License without being bound by section 3 of the GNU GPL. 2. Conveying Modified Versions. If you modify a copy of the Library, and, in your modifications, a facility refers to a function or data to be supplied by an Application that uses the facility (other than as an argument passed when the facility is invoked), then you may convey a copy of the modified version: a) under this License, provided that you make a good faith effort to ensure that, in the event an Application does not supply the function or data, the facility still operates, and performs whatever part of its purpose remains meaningful, or b) under the GNU GPL, with none of the additional permissions of this License applicable to that copy. 3. Object Code Incorporating Material from Library Header Files. The object code form of an Application may incorporate material from a header file that is part of the Library. You may convey such object code under terms of your choice, provided that, if the incorporated material is not limited to numerical parameters, data structure layouts and accessors, or small macros, inline functions and templates (ten or fewer lines in length), you do both of the following: a) Give prominent notice with each copy of the object code that the Library is used in it and that the Library and its use are covered by this License. b) Accompany the object code with a copy of the GNU GPL and this license document. 4. Combined Works. You may convey a Combined Work under terms of your choice that, taken together, effectively do not restrict modification of the portions of the Library contained in the Combined Work and reverse engineering for debugging such modifications, if you also do each of the following: a) Give prominent notice with each copy of the Combined Work that the Library is used in it and that the Library and its use are covered by this License. b) Accompany the Combined Work with a copy of the GNU GPL and this license document. c) For a Combined Work that displays copyright notices during execution, include the copyright notice for the Library among these notices, as well as a reference directing the user to the copies of the GNU GPL and this license document. d) Do one of the following: 0) Convey the Minimal Corresponding Source under the terms of this License, and the Corresponding Application Code in a form suitable for, and under terms that permit, the user to recombine or relink the Application with a modified version of the Linked Version to produce a modified Combined Work, in the manner specified by section 6 of the GNU GPL for conveying Corresponding Source. 1) Use a suitable shared library mechanism for linking with the Library. A suitable mechanism is one that (a) uses at run time a copy of the Library already present on the user's computer system, and (b) will operate properly with a modified version of the Library that is interface-compatible with the Linked Version. e) Provide Installation Information, but only if you would otherwise be required to provide such information under section 6 of the GNU GPL, and only to the extent that such information is necessary to install and execute a modified version of the Combined Work produced by recombining or relinking the Application with a modified version of the Linked Version. (If you use option 4d0, the Installation Information must accompany the Minimal Corresponding Source and Corresponding Application Code. If you use option 4d1, you must provide the Installation Information in the manner specified by section 6 of the GNU GPL for conveying Corresponding Source.) 5. Combined Libraries. You may place library facilities that are a work based on the Library side by side in a single library together with other library facilities that are not Applications and are not covered by this License, and convey such a combined library under terms of your choice, if you do both of the following: a) Accompany the combined library with a copy of the same work based on the Library, uncombined with any other library facilities, conveyed under the terms of this License. b) Give prominent notice with the combined library that part of it is a work based on the Library, and explaining where to find the accompanying uncombined form of the same work. 6. Revised Versions of the GNU Lesser General Public License. The Free Software Foundation may publish revised and/or new versions of the GNU Lesser 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 Library as you received it specifies that a certain numbered version of the GNU Lesser General Public License "or any later version" applies to it, you have the option of following the terms and conditions either of that published version or of any later version published by the Free Software Foundation. If the Library as you received it does not specify a version number of the GNU Lesser General Public License, you may choose any version of the GNU Lesser General Public License ever published by the Free Software Foundation. If the Library as you received it specifies that a proxy can decide whether future versions of the GNU Lesser General Public License shall apply, that proxy's public statement of acceptance of any version is permanent authorization for you to choose that version for the Library. cssutils-1.0/setup.cfg0000666000175000017500000000010012253307530013525 0ustar hugohugo[egg_info] tag_build = tag_date = 0 tag_svn_revision = 0 cssutils-1.0/src/0000755000175000017500000000000012416502415012500 5ustar hugohugocssutils-1.0/src/cssutils/0000755000175000017500000000000012416502415014351 5ustar hugohugocssutils-1.0/src/cssutils/codec.py0000666000175000017500000000060212126054542016003 0ustar hugohugo#!/usr/bin/env python """Python codec for CSS.""" __docformat__ = 'restructuredtext' __author__ = 'Walter Doerwald' __version__ = '$Id: util.py 1114 2008-03-05 13:22:59Z cthedot $' import sys if sys.version_info < (3,): from _codec2 import * # for tests from _codec2 import _fixencoding else: from _codec3 import * # for tests from _codec3 import _fixencoding cssutils-1.0/src/cssutils/sac.py0000666000175000017500000004356112126054542015507 0ustar hugohugo#!/usr/bin/env python """A validating CSSParser""" __all__ = ['CSSParser'] __docformat__ = 'restructuredtext' __version__ = '$Id: parse.py 1754 2009-05-30 14:50:13Z cthedot $' import helper import codecs import errorhandler import os import tokenize2 import urllib import sys class ErrorHandler(object): """Basic class for CSS error handlers. This class class provides a default implementation ignoring warnings and recoverable errors and throwing a SAXParseException for fatal errors. If a CSS application needs to implement customized error handling, it must extend this class and then register an instance with the CSS parser using the parser's setErrorHandler method. The parser will then report all errors and warnings through this interface. The parser shall use this class instead of throwing an exception: it is up to the application whether to throw an exception for different types of errors and warnings. Note, however, that there is no requirement that the parser continue to provide useful information after a call to fatalError (in other words, a CSS driver class could catch an exception and report a fatalError). """ def __init__(self): self._log = errorhandler.ErrorHandler() def error(self, exception, token=None): self._log.error(exception, token, neverraise=True) def fatal(self, exception, token=None): self._log.fatal(exception, token) def warn(self, exception, token=None): self._log.warn(exception, token, neverraise=True) class DocumentHandler(object): """ void endFontFace() Receive notification of the end of a font face statement. void endMedia(SACMediaList media) Receive notification of the end of a media statement. void endPage(java.lang.String name, java.lang.String pseudo_page) Receive notification of the end of a media statement. void importStyle(java.lang.String uri, SACMediaList media, java.lang.String defaultNamespaceURI) Receive notification of a import statement in the style sheet. void startFontFace() Receive notification of the beginning of a font face statement. void startMedia(SACMediaList media) Receive notification of the beginning of a media statement. void startPage(java.lang.String name, java.lang.String pseudo_page) Receive notification of the beginning of a page statement. """ def __init__(self): def log(msg): sys.stderr.write('INFO\t%s\n' % msg) self._log = log def comment(self, text, line=None, col=None): "Receive notification of a comment." self._log("comment %r at [%s, %s]" % (text, line, col)) def startDocument(self, encoding): "Receive notification of the beginning of a style sheet." # source self._log("startDocument encoding=%s" % encoding) def endDocument(self, source=None, line=None, col=None): "Receive notification of the end of a document." self._log("endDocument EOF") def importStyle(self, uri, media, name, line=None, col=None): "Receive notification of a import statement in the style sheet." # defaultNamespaceURI??? self._log("importStyle at [%s, %s]" % (line, col)) def namespaceDeclaration(self, prefix, uri, line=None, col=None): "Receive notification of an unknown rule t-rule not supported by this parser." # prefix might be None! self._log("namespaceDeclaration at [%s, %s]" % (line, col)) def startSelector(self, selectors=None, line=None, col=None): "Receive notification of the beginning of a rule statement." # TODO selectorList! self._log("startSelector at [%s, %s]" % (line, col)) def endSelector(self, selectors=None, line=None, col=None): "Receive notification of the end of a rule statement." self._log("endSelector at [%s, %s]" % (line, col)) def property(self, name, value='TODO', important=False, line=None, col=None): "Receive notification of a declaration." # TODO: value is LexicalValue? self._log("property %r at [%s, %s]" % (name, line, col)) def ignorableAtRule(self, atRule, line=None, col=None): "Receive notification of an unknown rule t-rule not supported by this parser." self._log("ignorableAtRule %r at [%s, %s]" % (atRule, line, col)) class EchoHandler(DocumentHandler): "Echos all input to property `out`" def __init__(self): super(EchoHandler, self).__init__() self._out = [] out = property(lambda self: u''.join(self._out)) def startDocument(self, encoding): super(EchoHandler, self).startDocument(encoding) if u'utf-8' != encoding: self._out.append(u'@charset "%s";\n' % encoding) # def comment(self, text, line=None, col=None): # self._out.append(u'/*%s*/' % text) def importStyle(self, uri, media, name, line=None, col=None): "Receive notification of a import statement in the style sheet." # defaultNamespaceURI??? super(EchoHandler, self).importStyle(uri, media, name, line, col) self._out.append(u'@import %s%s%s;\n' % (helper.string(uri), u'%s ' % media if media else u'', u'%s ' % name if name else u'') ) def namespaceDeclaration(self, prefix, uri, line=None, col=None): super(EchoHandler, self).namespaceDeclaration(prefix, uri, line, col) self._out.append(u'@namespace %s%s;\n' % (u'%s ' % prefix if prefix else u'', helper.string(uri))) def startSelector(self, selectors=None, line=None, col=None): super(EchoHandler, self).startSelector(selectors, line, col) if selectors: self._out.append(u', '.join(selectors)) self._out.append(u' {\n') def endSelector(self, selectors=None, line=None, col=None): self._out.append(u' }') def property(self, name, value, important=False, line=None, col=None): super(EchoHandler, self).property(name, value, line, col) self._out.append(u' %s: %s%s;\n' % (name, value, u' !important' if important else u'')) class Parser(object): """ java.lang.String getParserVersion() Returns a string about which CSS language is supported by this parser. boolean parsePriority(InputSource source) Parse a CSS priority value (e.g. LexicalUnit parsePropertyValue(InputSource source) Parse a CSS property value. void parseRule(InputSource source) Parse a CSS rule. SelectorList parseSelectors(InputSource source) Parse a comma separated list of selectors. void parseStyleDeclaration(InputSource source) Parse a CSS style declaration (without '{' and '}'). void parseStyleSheet(InputSource source) Parse a CSS document. void parseStyleSheet(java.lang.String uri) Parse a CSS document from a URI. void setConditionFactory(ConditionFactory conditionFactory) void setDocumentHandler(DocumentHandler handler) Allow an application to register a document event handler. void setErrorHandler(ErrorHandler handler) Allow an application to register an error event handler. void setLocale(java.util.Locale locale) Allow an application to request a locale for errors and warnings. void setSelectorFactory(SelectorFactory selectorFactory) """ def __init__(self, documentHandler=None, errorHandler=None): self._tokenizer = tokenize2.Tokenizer() if documentHandler: self.setDocumentHandler(documentHandler) else: self.setDocumentHandler(DocumentHandler()) if errorHandler: self.setErrorHandler(errorHandler) else: self.setErrorHandler(ErrorHandler()) def parseString(self, cssText, encoding=None): if isinstance(cssText, str): cssText = codecs.getdecoder('css')(cssText, encoding=encoding)[0] tokens = self._tokenizer.tokenize(cssText, fullsheet=True) def COMMENT(val, line, col): self._handler.comment(val[2:-2], line, col) def EOF(val, line, col): self._handler.endDocument(val, line, col) def simple(t): map = {'COMMENT': COMMENT, 'S': lambda val, line, col: None, 'EOF': EOF} type_, val, line, col = t if type_ in map: map[type_](val, line, col) return True else: return False # START PARSING t = tokens.next() type_, val, line, col = t encoding = 'utf-8' if 'CHARSET_SYM' == type_: # @charset "encoding"; # S encodingtoken = tokens.next() semicolontoken = tokens.next() if 'STRING' == type_: encoding = helper.stringvalue(val) # ; if 'STRING' == encodingtoken[0] and semicolontoken: encoding = helper.stringvalue(encodingtoken[1]) else: self._errorHandler.fatal(u'Invalid @charset') t = tokens.next() type_, val, line, col = t self._handler.startDocument(encoding) while True: start = (line, col) try: if simple(t): pass elif 'ATKEYWORD' == type_ or type_ in ('PAGE_SYM', 'MEDIA_SYM', 'FONT_FACE_SYM'): atRule = [val] braces = 0 while True: # read till end ; # TODO: or {} t = tokens.next() type_, val, line, col = t atRule.append(val) if u';' == val and not braces: break elif u'{' == val: braces += 1 elif u'}' == val: braces -= 1 if braces == 0: break self._handler.ignorableAtRule(u''.join(atRule), *start) elif 'IMPORT_SYM' == type_: # import URI or STRING media? name? uri, media, name = None, None, None while True: t = tokens.next() type_, val, line, col = t if 'STRING' == type_: uri = helper.stringvalue(val) elif 'URI' == type_: uri = helper.urivalue(val) elif u';' == val: break if uri: self._handler.importStyle(uri, media, name) else: self._errorHandler.error(u'Invalid @import' u' declaration at %r' % (start,)) elif 'NAMESPACE_SYM' == type_: prefix, uri = None, None while True: t = tokens.next() type_, val, line, col = t if 'IDENT' == type_: prefix = val elif 'STRING' == type_: uri = helper.stringvalue(val) elif 'URI' == type_: uri = helper.urivalue(val) elif u';' == val: break if uri: self._handler.namespaceDeclaration(prefix, uri, *start) else: self._errorHandler.error(u'Invalid @namespace' u' declaration at %r' % (start,)) else: # CSSSTYLERULE selector = [] selectors = [] while True: # selectors[, selector]* { if 'S' == type_: selector.append(u' ') elif simple(t): pass elif u',' == val: selectors.append(u''.join(selector).strip()) selector = [] elif u'{' == val: selectors.append(u''.join(selector).strip()) self._handler.startSelector(selectors, *start) break else: selector.append(val) t = tokens.next() type_, val, line, col = t end = None while True: # name: value [!important][;name: value [!important]]*;? name, value, important = None, [], False while True: # name: t = tokens.next() type_, val, line, col = t if 'S' == type_: pass elif simple(t): pass elif 'IDENT' == type_: if name: self._errorHandler.error('more than one property name', t) else: name = val elif u':' == val: if not name: self._errorHandler.error('no property name', t) break elif u';' == val: self._errorHandler.error('premature end of property', t) end = val break elif u'}' == val: if name: self._errorHandler.error('premature end of property', t) end = val break else: self._errorHandler.error('unexpected property name token %r' % val, t) while not u';' == end and not u'}' == end: # value !;} t = tokens.next() type_, val, line, col = t if 'S' == type_: value.append(u' ') elif simple(t): pass elif u'!' == val or u';' == val or u'}' == val: value = ''.join(value).strip() if not value: self._errorHandler.error('premature end of property (no value)', t) end = val break else: value.append(val) while u'!' == end: # !important t = tokens.next() type_, val, line, col = t if simple(t): pass elif u'IDENT' == type_ and not important: important = True elif u';' == val or u'}' == val: end = val break else: self._errorHandler.error('unexpected priority token %r' % val) if name and value: self._handler.property(name, value, important) if u'}' == end: self._handler.endSelector(selectors, line=line, col=col) break else: # reset end = None else: self._handler.endSelector(selectors, line=line, col=col) t = tokens.next() type_, val, line, col = t except StopIteration: break def setDocumentHandler(self, handler): "Allow an application to register a document event `handler`." self._handler = handler def setErrorHandler(self, handler): "TODO" self._errorHandler = handler cssutils-1.0/src/cssutils/_fetch.py0000666000175000017500000000327512250170414016162 0ustar hugohugo"""Default URL reading functions""" __all__ = ['_defaultFetcher'] __docformat__ = 'restructuredtext' __version__ = '$Id: tokenize2.py 1547 2008-12-10 20:42:26Z cthedot $' import cssutils from cssutils import VERSION import encutils import errorhandler import urllib2 log = errorhandler.ErrorHandler() def _defaultFetcher(url): """Retrieve data from ``url``. cssutils default implementation of fetch URL function. Returns ``(encoding, string)`` or ``None`` """ try: request = urllib2.Request(url) request.add_header('User-agent', 'cssutils %s (http://www.cthedot.de/cssutils/)' % VERSION) res = urllib2.urlopen(request) except urllib2.HTTPError, e: # http error, e.g. 404, e can be raised log.warn(u'HTTPError opening url=%s: %s %s' % (url, e.code, e.msg), error=e) except urllib2.URLError, e: # URLError like mailto: or other IO errors, e can be raised log.warn(u'URLError, %s' % e.reason, error=e) except OSError, e: # e.g if file URL and not found log.warn(e, error=OSError) except ValueError, e: # invalid url, e.g. "1" log.warn(u'ValueError, %s' % e.args[0], error=ValueError) else: if res: mimeType, encoding = encutils.getHTTPInfo(res) if mimeType != u'text/css': log.error(u'Expected "text/css" mime type for url=%r but found: %r' % (url, mimeType), error=ValueError) content = res.read() if hasattr(res, 'close'): res.close() return encoding, content cssutils-1.0/src/cssutils/util.py0000666000175000017500000007377212253134142015721 0ustar hugohugo"""base classes and helper functions for css and stylesheets packages """ __all__ = [] __docformat__ = 'restructuredtext' __version__ = '$Id$' from helper import normalize from itertools import ifilter, chain import cssutils import codec import codecs import errorhandler import tokenize2 import types import xml.dom try: from _fetchgae import _defaultFetcher except ImportError, e: from _fetch import _defaultFetcher log = errorhandler.ErrorHandler() class _BaseClass(object): """ Base class for Base, Base2 and _NewBase. **Base and Base2 will be removed in the future!** """ _log = errorhandler.ErrorHandler() _prods = tokenize2.CSSProductions def _checkReadonly(self): "Raise xml.dom.NoModificationAllowedErr if rule/... is readonly" if hasattr(self, '_readonly') and self._readonly: raise xml.dom.NoModificationAllowedErr( u'%s is readonly.' % self.__class__) return True return False def _valuestr(self, t): """ Return string value of t (t may be a string, a list of token tuples or a single tuple in format (type, value, line, col). Mainly used to get a string value of t for error messages. """ if not t: return u'' elif isinstance(t, basestring): return t else: return u''.join([x[1] for x in t]) class _NewBase(_BaseClass): """ New base class for classes using ProdParser. **Currently CSSValue and related ones only.** """ def __init__(self): self._seq = Seq() def _setSeq(self, newseq): """Set value of ``seq`` which is readonly.""" newseq._readonly = True self._seq = newseq def _clearSeq(self): self._seq.clear() def _tempSeq(self, readonly=False): "Get a writeable Seq() which is used to set ``seq`` later" return Seq(readonly=readonly) seq = property(lambda self: self._seq, doc="Internal readonly attribute, **DO NOT USE**!") class _NewListBase(_NewBase): """ (EXPERIMENTAL) A base class used for list classes like stylesheets.MediaList adds list like behaviour running on inhering class' property ``seq`` - item in x => bool - len(x) => integer - get, set and del x[i] - for item in x - append(item) some methods must be overwritten in inheriting class """ def __init__(self): self._seq = Seq() def __contains__(self, item): for it in self._seq: if item == it.value: return True return False def __delitem__(self, index): del self._seq[index] def __getitem__(self, index): return self._seq[index].value def __iter__(self): def gen(): for x in self._seq: yield x.value return gen() def __len__(self): return len(self._seq) def __setitem__(self, index, item): "must be overwritten" raise NotImplementedError def append(self, item): "must be overwritten" raise NotImplementedError class Base(_BaseClass): """ **Superceded by _NewBase** **Superceded by Base2 which is used for new seq handling class.** Base class for most CSS and StyleSheets classes Contains helper methods for inheriting classes helping parsing ``_normalize`` is static as used by Preferences. """ __tokenizer2 = tokenize2.Tokenizer() # for more on shorthand properties see # http://www.dustindiaz.com/css-shorthand/ # format: shorthand: [(propname, mandatorycheck?)*] _SHORTHANDPROPERTIES = { u'background': [], #u'background-position': [], # list of 2 values! u'border': [], u'border-left': [], u'border-right': [], u'border-top': [], u'border-bottom': [], #u'border-color': [], # list or single but same values #u'border-style': [], # list or single but same values #u'border-width': [], # list or single but same values u'cue': [], u'font': [], u'list-style': [], #u'margin': [], # list or single but same values u'outline': [], #u'padding': [], # list or single but same values u'pause': [] } @staticmethod def _normalize(x): """ normalizes x, namely: - remove any \ before non unicode sequences (0-9a-zA-Z) so for x=="c\olor\" return "color" (unicode escape sequences should have been resolved by the tokenizer already) - lowercase """ return normalize(x) def _splitNamespacesOff(self, text_namespaces_tuple): """ returns tuple (text, dict-of-namespaces) or if no namespaces are in cssText returns (cssText, {}) used in Selector, SelectorList, CSSStyleRule, CSSMediaRule and CSSStyleSheet """ if isinstance(text_namespaces_tuple, tuple): return text_namespaces_tuple[0], _SimpleNamespaces(self._log, text_namespaces_tuple[1]) else: return text_namespaces_tuple, _SimpleNamespaces(log=self._log) def _tokenize2(self, textortokens): """ returns tokens of textortokens which may already be tokens in which case simply returns input """ if not textortokens: return None elif isinstance(textortokens, basestring): # needs to be tokenized return self.__tokenizer2.tokenize( textortokens) elif isinstance(textortokens, tuple): # a single token (like a comment) return [textortokens] else: # already tokenized but return an iterator return iter(textortokens) def _nexttoken(self, tokenizer, default=None): "returns next token in generator tokenizer or the default value" try: return tokenizer.next() # TypeError for py3 except (StopIteration, AttributeError, TypeError): return default def _type(self, token): "returns type of Tokenizer token" if token: return token[0] else: return None def _tokenvalue(self, token, normalize=False): "returns value of Tokenizer token" if token and normalize: return Base._normalize(token[1]) elif token: return token[1] else: return None def _stringtokenvalue(self, token): """ for STRING returns the actual content without surrounding "" or '' and without respective escapes, e.g.:: "with \" char" => with " char """ if token: value = token[1] return value.replace('\\' + value[0], value[0])[1: - 1] else: return None def _uritokenvalue(self, token): """ for URI returns the actual content without surrounding url() or url(""), url('') and without respective escapes, e.g.:: url("\"") => " """ if token: value = token[1][4: - 1].strip() if value and (value[0] in '\'"') and (value[0] == value[ - 1]): # a string "..." or '...' value = value.replace('\\' + value[0], value[0])[1: - 1] return value else: return None def _tokensupto2(self, tokenizer, starttoken=None, blockstartonly=False, # { blockendonly=False, # } mediaendonly=False, importmediaqueryendonly=False, # ; or STRING mediaqueryendonly=False, # { or STRING semicolon=False, # ; propertynameendonly=False, # : propertyvalueendonly=False, # ! ; } propertypriorityendonly=False, # ; } selectorattendonly=False, # ] funcendonly=False, # ) listseponly=False, # , separateEnd=False # returns (resulttokens, endtoken) ): """ returns tokens upto end of atrule and end index end is defined by parameters, might be ; } ) or other default looks for ending "}" and ";" """ ends = u';}' endtypes = () brace = bracket = parant = 0 # {}, [], () if blockstartonly: # { ends = u'{' brace = - 1 # set to 0 with first { elif blockendonly: # } ends = u'}' brace = 1 elif mediaendonly: # } ends = u'}' brace = 1 # rules } and mediarules } elif importmediaqueryendonly: # end of mediaquery which may be ; or STRING ends = u';' endtypes = ('STRING',) elif mediaqueryendonly: # end of mediaquery which may be { or STRING # special case, see below ends = u'{' brace = - 1 # set to 0 with first { endtypes = ('STRING',) elif semicolon: ends = u';' elif propertynameendonly: # : and ; in case of an error ends = u':;' elif propertyvalueendonly: # ; or !important ends = u';!' elif propertypriorityendonly: # ; ends = u';' elif selectorattendonly: # ] ends = u']' if starttoken and self._tokenvalue(starttoken) == u'[': bracket = 1 elif funcendonly: # ) ends = u')' parant = 1 elif listseponly: # , ends = u',' resulttokens = [] if starttoken: resulttokens.append(starttoken) val = starttoken[1] if u'[' == val: bracket += 1 elif u'{' == val: brace += 1 elif u'(' == val: parant += 1 if tokenizer: for token in tokenizer: typ, val, line, col = token if 'EOF' == typ: resulttokens.append(token) break if u'{' == val: brace += 1 elif u'}' == val: brace -= 1 elif u'[' == val: bracket += 1 elif u']' == val: bracket -= 1 # function( or single ( elif u'(' == val or \ Base._prods.FUNCTION == typ: parant += 1 elif u')' == val: parant -= 1 resulttokens.append(token) if (brace == bracket == parant == 0) and ( val in ends or typ in endtypes): break elif mediaqueryendonly and brace == - 1 and ( bracket == parant == 0) and typ in endtypes: # mediaqueryendonly with STRING break if separateEnd: # TODO: use this method as generator, then this makes sense if resulttokens: return resulttokens[: - 1], resulttokens[ - 1] else: return resulttokens, None else: return resulttokens def _adddefaultproductions(self, productions, new=None): """ adds default productions if not already present, used by _parse only each production should return the next expected token normaly a name like "uri" or "EOF" some have no expectation like S or COMMENT, so simply return the current value of self.__expected """ def ATKEYWORD(expected, seq, token, tokenizer=None): "default impl for unexpected @rule" if expected != 'EOF': # TODO: parentStyleSheet=self rule = cssutils.css.CSSUnknownRule() rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: seq.append(rule) return expected else: new['wellformed'] = False self._log.error(u'Expected EOF.', token=token) return expected def COMMENT(expected, seq, token, tokenizer=None): "default implementation for COMMENT token adds CSSCommentRule" seq.append(cssutils.css.CSSComment([token])) return expected def S(expected, seq, token, tokenizer=None): "default implementation for S token, does nothing" return expected def EOF(expected=None, seq=None, token=None, tokenizer=None): "default implementation for EOF token" return 'EOF' p = {'ATKEYWORD': ATKEYWORD, 'COMMENT': COMMENT, 'S': S, 'EOF': EOF # only available if fullsheet } p.update(productions) return p def _parse(self, expected, seq, tokenizer, productions, default=None, new=None, initialtoken=None): """ puts parsed tokens in seq by calling a production with (seq, tokenizer, token) expected a name what token or value is expected next, e.g. 'uri' seq to add rules etc to tokenizer call tokenizer.next() to get next token productions callbacks {tokentype: callback} default default callback if tokentype not in productions new used to init default productions initialtoken will be used together with tokenizer running 1st this token and then all tokens in tokenizer returns (wellformed, expected) which the last prod might have set """ wellformed = True if initialtoken: # add initialtoken to tokenizer def tokens(): "Build new tokenizer including initialtoken" yield initialtoken for item in tokenizer: yield item fulltokenizer = chain([initialtoken], tokenizer) else: fulltokenizer = tokenizer if fulltokenizer: prods = self._adddefaultproductions(productions, new) for token in fulltokenizer: p = prods.get(token[0], default) if p: expected = p(expected, seq, token, tokenizer) else: wellformed = False self._log.error(u'Unexpected token (%s, %s, %s, %s)' % token) return wellformed, expected class Base2(Base, _NewBase): """ **Superceded by _NewBase.** Base class for new seq handling. """ def __init__(self): self._seq = Seq() def _adddefaultproductions(self, productions, new=None): """ adds default productions if not already present, used by _parse only each production should return the next expected token normaly a name like "uri" or "EOF" some have no expectation like S or COMMENT, so simply return the current value of self.__expected """ def ATKEYWORD(expected, seq, token, tokenizer=None): "default impl for unexpected @rule" if expected != 'EOF': # TODO: parentStyleSheet=self rule = cssutils.css.CSSUnknownRule() rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: seq.append(rule, cssutils.css.CSSRule.UNKNOWN_RULE, line=token[2], col=token[3]) return expected else: new['wellformed'] = False self._log.error(u'Expected EOF.', token=token) return expected def COMMENT(expected, seq, token, tokenizer=None): "default impl, adds CSSCommentRule if not token == EOF" if expected == 'EOF': new['wellformed'] = False self._log.error(u'Expected EOF but found comment.', token=token) seq.append(cssutils.css.CSSComment([token]), 'COMMENT') return expected def S(expected, seq, token, tokenizer=None): "default impl, does nothing if not token == EOF" if expected == 'EOF': new['wellformed'] = False self._log.error(u'Expected EOF but found whitespace.', token=token) return expected def EOF(expected=None, seq=None, token=None, tokenizer=None): "default implementation for EOF token" return 'EOF' defaultproductions = {'ATKEYWORD': ATKEYWORD, 'COMMENT': COMMENT, 'S': S, 'EOF': EOF # only available if fullsheet } defaultproductions.update(productions) return defaultproductions class Seq(object): """ property seq of Base2 inheriting classes, holds a list of Item objects. used only by Selector for now is normally readonly, only writable during parsing """ def __init__(self, readonly=True): """ only way to write to a Seq is to initialize it with new items each itemtuple has (value, type, line) where line is optional """ self._seq = [] self._readonly = readonly def __repr__(self): "returns a repr same as a list of tuples of (value, type)" return u'cssutils.%s.%s([\n %s], readonly=%r)' % (self.__module__, self.__class__.__name__, u',\n '.join([u'%r' % item for item in self._seq] ), self._readonly) def __str__(self): vals = [] for v in self: if isinstance(v.value, basestring): vals.append(v.value) elif isinstance(v, tuple): vals.append(v.value[1]) else: vals.append(repr(v)) return "" % ( self.__module__, self.__class__.__name__, len(self), vals, self._readonly, id(self)) def __delitem__(self, i): del self._seq[i] def __getitem__(self, i): return self._seq[i] def __setitem__(self, i, (val, typ, line, col)): self._seq[i] = Item(val, typ, line, col) def __iter__(self): return iter(self._seq) def __len__(self): return len(self._seq) def append(self, val, typ=None, line=None, col=None): "If not readonly add new Item()" if self._readonly: raise AttributeError('Seq is readonly.') else: if isinstance(val, Item): self._seq.append(val) else: self._seq.append(Item(val, typ, line, col)) def appendItem(self, item): "if not readonly add item which must be an Item" if self._readonly: raise AttributeError('Seq is readonly.') else: self._seq.append(item) def clear(self): del self._seq[:] def insert(self, index, val, typ, line=None, col=None): "Insert new Item() at index # even if readony!? TODO!" self._seq.insert(index, Item(val, typ, line, col)) def replace(self, index=-1, val=None, typ=None, line=None, col=None): """ if not readonly replace Item at index with new Item or simply replace value or type """ if self._readonly: raise AttributeError('Seq is readonly.') else: self._seq[index] = Item(val, typ, line, col) def rstrip(self): "trims S items from end of Seq" while self._seq and self._seq[ - 1].type == tokenize2.CSSProductions.S: # TODO: removed S before CSSComment /**/ /**/ del self._seq[ - 1] def appendToVal(self, val=None, index= - 1): """ if not readonly append to Item's value at index """ if self._readonly: raise AttributeError('Seq is readonly.') else: old = self._seq[index] self._seq[index] = Item(old.value + val, old.type, old.line, old.col) class Item(object): """ an item in the seq list of classes (successor to tuple items in old seq) each item has attributes: type a sematic type like "element", "attribute" value the actual value which may be a string, number etc or an instance of e.g. a CSSComment *line* **NOT IMPLEMENTED YET, may contain the line in the source later** """ def __init__(self, value, type, line=None, col=None): self.__value = value self.__type = type self.__line = line self.__col = col type = property(lambda self: self.__type) value = property(lambda self: self.__value) line = property(lambda self: self.__line) col = property(lambda self: self.__col) def __repr__(self): return "%s.%s(value=%r, type=%r, line=%r, col=%r)" % ( self.__module__, self.__class__.__name__, self.__value, self.__type, self.__line, self.__col) class ListSeq(object): """ (EXPERIMENTAL) A base class used for list classes like cssutils.css.SelectorList or stylesheets.MediaList adds list like behaviour running on inhering class' property ``seq`` - item in x => bool - len(x) => integer - get, set and del x[i] - for item in x - append(item) some methods must be overwritten in inheriting class """ def __init__(self): self.seq = [] # does not need to use ``Seq`` as simple list only def __contains__(self, item): return item in self.seq def __delitem__(self, index): del self.seq[index] def __getitem__(self, index): return self.seq[index] def __iter__(self): def gen(): for x in self.seq: yield x return gen() def __len__(self): return len(self.seq) def __setitem__(self, index, item): "must be overwritten" raise NotImplementedError def append(self, item): "must be overwritten" raise NotImplementedError class _Namespaces(object): """ A dictionary like wrapper for @namespace rules used in a CSSStyleSheet. Works on effective namespaces, so e.g. if:: @namespace p1 "uri"; @namespace p2 "uri"; only the second rule is effective and kept. namespaces a dictionary {prefix: namespaceURI} containing the effective namespaces only. These are the latest set in the CSSStyleSheet. parentStyleSheet the parent CSSStyleSheet """ def __init__(self, parentStyleSheet, log=None, *args): "no initial values are set, only the relevant sheet is" self.parentStyleSheet = parentStyleSheet self._log = log def __repr__(self): return "%r" % self.namespaces def __contains__(self, prefix): return prefix in self.namespaces def __delitem__(self, prefix): """deletes CSSNamespaceRule(s) with rule.prefix == prefix prefix '' and None are handled the same """ if not prefix: prefix = u'' delrule = self.__findrule(prefix) for i, rule in enumerate(ifilter(lambda r: r.type == r.NAMESPACE_RULE, self.parentStyleSheet.cssRules)): if rule == delrule: self.parentStyleSheet.deleteRule(i) return self._log.error('Prefix %s not found.' % prefix, error=xml.dom.NamespaceErr) def __getitem__(self, prefix): try: return self.namespaces[prefix] except KeyError, e: self._log.error('Prefix %s not found.' % prefix, error=xml.dom.NamespaceErr) def __iter__(self): return self.namespaces.__iter__() def __len__(self): return len(self.namespaces) def __setitem__(self, prefix, namespaceURI): "replaces prefix or sets new rule, may raise NoModificationAllowedErr" if not prefix: prefix = u'' # None or '' rule = self.__findrule(prefix) if not rule: self.parentStyleSheet.insertRule(cssutils.css.CSSNamespaceRule( prefix=prefix, namespaceURI=namespaceURI), inOrder=True) else: if prefix in self.namespaces: rule.namespaceURI = namespaceURI # raises NoModificationAllowedErr if namespaceURI in self.namespaces.values(): rule.prefix = prefix def __findrule(self, prefix): # returns namespace rule where prefix == key for rule in ifilter(lambda r: r.type == r.NAMESPACE_RULE, reversed(self.parentStyleSheet.cssRules)): if rule.prefix == prefix: return rule @property def namespaces(self): """ A property holding only effective @namespace rules in self.parentStyleSheets. """ namespaces = {} for rule in ifilter(lambda r: r.type == r.NAMESPACE_RULE, reversed(self.parentStyleSheet.cssRules)): if rule.namespaceURI not in namespaces.values(): namespaces[rule.prefix] = rule.namespaceURI return namespaces def get(self, prefix, default): return self.namespaces.get(prefix, default) def items(self): return self.namespaces.items() def keys(self): return self.namespaces.keys() def values(self): return self.namespaces.values() def prefixForNamespaceURI(self, namespaceURI): """ returns effective prefix for given namespaceURI or raises IndexError if this cannot be found""" for prefix, uri in self.namespaces.items(): if uri == namespaceURI: return prefix raise IndexError(u'NamespaceURI %s not found.' % namespaceURI) def __str__(self): return u"" % ( self.__class__.__name__, str(self.parentStyleSheet), id(self)) class _SimpleNamespaces(_Namespaces): """ namespaces used in objects like Selector as long as they are not connected to a CSSStyleSheet """ def __init__(self, log=None, *args): """init""" super(_SimpleNamespaces, self).__init__(parentStyleSheet=None, log=log) self.__namespaces = dict(*args) def __setitem__(self, prefix, namespaceURI): self.__namespaces[prefix] = namespaceURI namespaces = property(lambda self: self.__namespaces, doc=u'Dict Wrapper for self.sheets @namespace rules.') def __str__(self): return u"" % ( self.__class__.__name__, self.namespaces, id(self)) def __repr__(self): return u"cssutils.util.%s(%r)" % (self.__class__.__name__, self.namespaces) def _readUrl(url, fetcher=None, overrideEncoding=None, parentEncoding=None): """ Read cssText from url and decode it using all relevant methods (HTTP header, BOM, @charset). Returns - encoding used to decode text (which is needed to set encoding of stylesheet properly) - type of encoding (how it was retrieved, see list below) - decodedCssText ``fetcher`` see cssutils.CSSParser.setFetcher for details ``overrideEncoding`` If given this encoding is used and all other encoding information is ignored (HTTP, BOM etc) ``parentEncoding`` Encoding of parent stylesheet (while e.g. reading @import references sheets) or document if available. Priority or encoding information -------------------------------- **cssutils only**: 0. overrideEncoding 1. An HTTP "charset" parameter in a "Content-Type" field (or similar parameters in other protocols) 2. BOM and/or @charset (see below) 3. or other metadata from the linking mechanism (if any) 4. charset of referring style sheet or document (if any) 5. Assume UTF-8 """ enctype = None if not fetcher: fetcher = _defaultFetcher r = fetcher(url) if r and len(r) == 2 and r[1] is not None: httpEncoding, content = r if overrideEncoding: enctype = 0 # 0. override encoding encoding = overrideEncoding elif httpEncoding: enctype = 1 # 1. HTTP encoding = httpEncoding else: # BOM or @charset if isinstance(content, unicode): contentEncoding, explicit = codec.detectencoding_unicode(content) else: contentEncoding, explicit = codec.detectencoding_str(content) if explicit: enctype = 2 # 2. BOM/@charset: explicitly encoding = contentEncoding elif parentEncoding: enctype = 4 # 4. parent stylesheet or document # may also be None in which case 5. is used in next step anyway encoding = parentEncoding else: enctype = 5 # 5. assume UTF-8 encoding = 'utf-8' if isinstance(content, unicode): decodedCssText = content else: try: # encoding may still be wrong if encoding *is lying*! try: decodedCssText = codecs.lookup("css")[1](content, encoding=encoding)[0] except AttributeError, ae: # at least in GAE decodedCssText = content.decode(encoding if encoding else 'utf-8') except UnicodeDecodeError, e: log.warn(e, neverraise=True) decodedCssText = None return encoding, enctype, decodedCssText else: return None, None, None cssutils-1.0/src/cssutils/_fetchgae.py0000666000175000017500000000512312126054542016636 0ustar hugohugo"""GAE specific URL reading functions""" __all__ = ['_defaultFetcher'] __docformat__ = 'restructuredtext' __version__ = '$Id: tokenize2.py 1547 2008-12-10 20:42:26Z cthedot $' # raises ImportError of not on GAE from google.appengine.api import urlfetch import cgi import errorhandler import util log = errorhandler.ErrorHandler() def _defaultFetcher(url): """ uses GoogleAppEngine (GAE) fetch(url, payload=None, method=GET, headers={}, allow_truncated=False) Response content The body content of the response. content_was_truncated True if the allow_truncated parameter to fetch() was True and the response exceeded the maximum response size. In this case, the content attribute contains the truncated response. status_code The HTTP status code. headers The HTTP response headers, as a mapping of names to values. Exceptions exception InvalidURLError() The URL of the request was not a valid URL, or it used an unsupported method. Only http and https URLs are supported. exception DownloadError() There was an error retrieving the data. This exception is not raised if the server returns an HTTP error code: In that case, the response data comes back intact, including the error code. exception ResponseTooLargeError() The response data exceeded the maximum allowed size, and the allow_truncated parameter passed to fetch() was False. """ #from google.appengine.api import urlfetch try: r = urlfetch.fetch(url, method=urlfetch.GET) except urlfetch.Error, e: log.warn(u'Error opening url=%r: %s' % (url, e), error=IOError) else: if r.status_code == 200: # find mimetype and encoding mimetype = 'application/octet-stream' try: mimetype, params = cgi.parse_header(r.headers['content-type']) encoding = params['charset'] except KeyError: encoding = None if mimetype != u'text/css': log.error(u'Expected "text/css" mime type for url %r but found: %r' % (url, mimetype), error=ValueError) return encoding, r.content else: # TODO: 301 etc log.warn(u'Error opening url=%r: HTTP status %s' % (url, r.status_code), error=IOError) cssutils-1.0/src/cssutils/prodparser.py0000666000175000017500000007057312253153760017130 0ustar hugohugo# -*- coding: utf-8 -*- """Productions parser used by css and stylesheets classes to parse test into a cssutils.util.Seq and at the same time retrieving additional specific cssutils.util.Item objects for later use. TODO: - ProdsParser - handle EOF or STOP? - handle unknown @rules - handle S: maybe save to Seq? parameterized? - store['_raw']: always? - Sequence: - opt first(), naive impl for now """ __all__ = ['ProdParser', 'Sequence', 'Choice', 'Prod', 'PreDef'] __docformat__ = 'restructuredtext' __version__ = '$Id: parse.py 1418 2008-08-09 19:27:50Z cthedot $' from helper import pushtoken import cssutils import itertools import re import string import sys import types class ParseError(Exception): """Base Exception class for ProdParser (used internally).""" pass class Done(ParseError): """Raised if Sequence or Choice is finished and no more Prods left.""" pass class Exhausted(ParseError): """Raised if Sequence or Choice is finished but token is given.""" pass class Missing(ParseError): """Raised if Sequence or Choice is not finished but no matching token given.""" pass class NoMatch(ParseError): """Raised if nothing in Sequence or Choice does match.""" pass class Choice(object): """A Choice of productions (Sequence or single Prod).""" def __init__(self, *prods, **options): """ *prods Prod or Sequence objects options: optional=False """ self._prods = prods try: self.optional = options['optional'] except KeyError, e: for p in self._prods: if p.optional: self.optional = True break else: self.optional = False self.reset() def reset(self): """Start Choice from zero""" self._exhausted = False def matches(self, token): """Check if token matches""" for prod in self._prods: if prod.matches(token): return True return False def nextProd(self, token): """ Return: - next matching Prod or Sequence - ``None`` if any Prod or Sequence is optional and no token matched - raise ParseError if nothing matches and all are mandatory - raise Exhausted if choice already done ``token`` may be None but this occurs when no tokens left.""" #print u'TEST for %s in %s' % (token, self) if not self._exhausted: optional = False for p in self._prods: if p.matches(token): self._exhausted = True p.reset() #print u'FOUND for %s: %s' % (token, p);#print return p elif p.optional: optional = True else: if not optional: # None matched but also None is optional raise NoMatch(u'No match for %s in %s' % (token, self)) #raise ParseError(u'No match in %s for %s' % (self, token)) elif token: raise Exhausted(u'Extra token') def __repr__(self): return "" % ( self.__class__.__name__, self.__str__(), self.optional, id(self)) def __str__(self): return u'Choice(%s)' % u', '.join([str(x) for x in self._prods]) class Sequence(object): """A Sequence of productions (Choice or single Prod).""" def __init__(self, *prods, **options): """ *prods Prod or Choice or Sequence objects **options: minmax = lambda: (1, 1) callback returning number of times this sequence may run """ self._prods = prods try: minmax = options['minmax'] except KeyError: minmax = lambda: (1, 1) self._min, self._max = minmax() if self._max is None: # unlimited try: # py2.6/3 self._max = sys.maxsize except AttributeError: # py<2.6 self._max = sys.maxint self._prodcount = len(self._prods) self.reset() def matches(self, token): """Called by Choice to try to find if Sequence matches.""" for prod in self._prods: if prod.matches(token): return True try: if not prod.optional: break except AttributeError: pass return False def reset(self): """Reset this Sequence if it is nested.""" self._roundstarted = False self._i = 0 self._round = 0 def _currentName(self): """Return current element of Sequence, used by name""" # TODO: current impl first only if 1st if an prod! for prod in self._prods[self._i:]: if not prod.optional: return str(prod) else: return 'Sequence' optional = property(lambda self: self._min == 0) def nextProd(self, token): """Return - next matching Prod or Choice - raises ParseError if nothing matches - raises Exhausted if sequence already done """ #print u'TEST for %s in %s' % (token, self) while self._round < self._max: # for this round i = self._i round = self._round p = self._prods[i] if i == 0: self._roundstarted = False # for next round self._i += 1 if self._i == self._prodcount: self._round += 1 self._i = 0 if p.matches(token): self._roundstarted = True # reset nested Choice or Prod to use from start p.reset() #print u'FOUND for %s: %s' % (token, p);#print return p elif p.optional: continue elif round < self._min or self._roundstarted: #or (round == 0 and self._min == 0): raise Missing(u'Missing token for production %s' % p) elif not token: if self._roundstarted: raise Missing(u'Missing token for production %s' % p) else: raise Done() else: raise NoMatch(u'No match for %s in %s' % (token, self)) if token: raise Exhausted(u'Extra token') def __repr__(self): return "" % ( self.__class__.__name__, self.__str__(), self.optional, id(self)) def __str__(self): return u'Sequence(%s)' % u', '.join([str(x) for x in self._prods]) class Prod(object): """Single Prod in Sequence or Choice.""" def __init__(self, name, match, optional=False, toSeq=None, toStore=None, stop=False, stopAndKeep=False, stopIfNoMoreMatch=False, nextSor=False, mayEnd=False, storeToken=None, exception=None): """ name name used for error reporting match callback function called with parameters tokentype and tokenvalue returning True, False or raising ParseError toSeq callback (optional) or False calling toSeq(token, tokens) returns (type_, val) == (token[0], token[1]) to be appended to seq else simply unaltered (type_, val) if False nothing is added toStore (optional) key to save util.Item to store or callback(store, util.Item) optional = False whether Prod is optional or not stop = False if True stop parsing of tokens here stopAndKeep if True stop parsing of tokens here but return stopping token in unused tokens stopIfNoMoreMatch = False stop even if more tokens available, similar to stop and keep but with condition no more matches nextSor=False next is S or other like , or / (CSSValue) mayEnd = False no token must follow even defined by Sequence. Used for operator ',/ ' currently only storeToken = None if True toStore saves simple token tuple and not and Item object to store. Old style processing, TODO: resolve exception = None exception to be raised in case of error, normaly SyntaxErr """ self._name = name self.match = match self.optional = optional self.stop = stop self.stopAndKeep = stopAndKeep self.stopIfNoMoreMatch = stopIfNoMoreMatch self.nextSor = nextSor self.mayEnd = mayEnd self.storeToken = storeToken self.exception = exception def makeToStore(key): "Return a function used by toStore." def toStore(store, item): "Set or append store item." if key in store: _v = store[key] if not isinstance(_v, list): store[key] = [_v] store[key].append(item) else: store[key] = item return toStore if toSeq or toSeq is False: # called: seq.append(toSeq(value)) self.toSeq = toSeq else: self.toSeq = lambda t, tokens: (t[0], t[1]) if hasattr(toStore, '__call__'): self.toStore = toStore elif toStore: self.toStore = makeToStore(toStore) else: # always set! self.toStore = None def matches(self, token): """Return if token matches.""" if not token: return False type_, val, line, col = token return self.match(type_, val) def reset(self): pass def __str__(self): return self._name def __repr__(self): return "" % ( self.__class__.__name__, self._name, id(self)) # global tokenizer as there is only one! tokenizer = cssutils.tokenize2.Tokenizer() # global: saved from subProds savedTokens = [] class ProdParser(object): """Productions parser.""" def __init__(self, clear=True): self.types = cssutils.cssproductions.CSSProductions self._log = cssutils.log if clear: tokenizer.clear() def _texttotokens(self, text): """Build a generator which is the only thing that is parsed! old classes may use lists etc """ if isinstance(text, basestring): # DEFAULT, to tokenize strip space return tokenizer.tokenize(text.strip()) elif type(text) == types.GeneratorType: # DEFAULT, already tokenized, should be generator return text elif isinstance(text, tuple): # OLD: (token, tokens) or a single token if len(text) == 2: # (token, tokens) chain([token], tokens) else: # single token return iter([text]) elif isinstance(text, list): # OLD: generator from list return iter(text) else: # ? return text def _SorTokens(self, tokens, until=',/'): """New tokens generator which has S tokens removed, if followed by anything in ``until``, normally a ``,``.""" for token in tokens: if token[0] == self.types.S: try: next_ = tokens.next() except StopIteration: yield token else: if next_[1] in until: # omit S as e.g. ``,`` has been found yield next_ elif next_[0] == self.types.COMMENT: # pass COMMENT yield next_ else: yield token yield next_ elif token[0] == self.types.COMMENT: # pass COMMENT yield token else: yield token break # normal mode again for token in tokens: yield token def parse(self, text, name, productions, keepS=False, checkS=False, store=None, emptyOk=False, debug=False): """ text (or token generator) to parse, will be tokenized if not a generator yet may be: - a string to be tokenized - a single token, a tuple - a tuple of (token, tokensGenerator) - already tokenized so a tokens generator name used for logging productions used to parse tokens keepS if WS should be added to Seq or just be ignored store UPDATED If a Prod defines ``toStore`` the key defined there is a key in store to be set or if store[key] is a list the next Item is appended here. TODO: NEEDED? : Key ``raw`` is always added and holds all unprocessed values found emptyOk if True text may be empty, hard to test before as may be generator returns :wellformed: True or False :seq: a filled cssutils.util.Seq object which is NOT readonly yet :store: filled keys defined by Prod.toStore :unusedtokens: token generator containing tokens not used yet """ tokens = self._texttotokens(text) if not tokens: self._log.error(u'No content to parse.') return False, [], None, None seq = cssutils.util.Seq(readonly=False) if not store: # store for specific values store = {} prods = [productions] # stack of productions wellformed = True # while no real token is found any S are ignored started = False stopall = False prod = None # flag if default S handling should be done defaultS = True stopIfNoMoreMatch = False stopIfNoMoreMatchNow = False while True: # get from savedTokens or normal tokens try: #print debug, "SAVED", savedTokens token = savedTokens.pop() except IndexError, e: try: token = tokens.next() except StopIteration: break #print debug, token, stopIfNoMoreMatch type_, val, line, col = token # default productions if type_ == self.types.COMMENT: # always append COMMENT seq.append(cssutils.css.CSSComment(val), cssutils.css.CSSComment, line, col) elif defaultS and type_ == self.types.S and not checkS: # append S (but ignore starting ones) if not keepS or not started: continue else: seq.append(val, type_, line, col) # elif type_ == self.types.ATKEYWORD: # # @rule # r = cssutils.css.CSSUnknownRule(cssText=val) # seq.append(r, type(r), line, col) elif type_ == self.types.INVALID: # invalidate parse wellformed = False self._log.error(u'Invalid token: %r' % (token,)) break elif type_ == 'EOF': # do nothing? (self.types.EOF == True!) stopall = True else: started = True # check S now nextSor = False # reset try: while True: # find next matching production try: prod = prods[-1].nextProd(token) except (Exhausted, NoMatch), e: # try next prod = None if isinstance(prod, Prod): # found actual Prod, not a Choice or Sequence break elif prod: # nested Sequence, Choice prods.append(prod) else: # nested exhausted, try in parent if len(prods) > 1: prods.pop() else: raise NoMatch('No match') except NoMatch, e: if stopIfNoMoreMatch: # and token: #print "\t1stopIfNoMoreMatch", e, token, prod, 'PUSHING' #tokenizer.push(token) savedTokens.append(token) stopIfNoMoreMatchNow = True stopall = True else: wellformed = False self._log.error(u'%s: %s: %r' % (name, e, token)) break except ParseError, e: # needed??? if stopIfNoMoreMatch: # and token: #print "\t2stopIfNoMoreMatch", e, token, prod tokenizer.push(token) stopIfNoMoreMatchNow = True stopall = True else: wellformed = False self._log.error(u'%s: %s: %r' % (name, e, token)) break else: #print '\t1', debug, 'PROD', prod # may stop next time, once set stays stopIfNoMoreMatch = prod.stopIfNoMoreMatch or stopIfNoMoreMatch # process prod if prod.toSeq and not prod.stopAndKeep: type_, val = prod.toSeq(token, tokens) if val is not None: seq.append(val, type_, line, col) if prod.toStore: if not prod.storeToken: prod.toStore(store, seq[-1]) else: # workaround for now for old style token # parsing! # TODO: remove when all new style prod.toStore(store, token) if prod.stop: # stop here and ignore following tokens # EOF? or end of e.g. func ")" break if prod.stopAndKeep: # e.g. ; # stop here and ignore following tokens # but keep this token for next run # TODO: CHECK!!!! tokenizer.push(token) tokens = itertools.chain(token, tokens) stopall = True break if prod.nextSor: # following is S or other token (e.g. ",")? # remove S if tokens = self._SorTokens(tokens, ',/') defaultS = False else: defaultS = True lastprod = prod #print debug, 'parse done', token, stopall, '\n' if not stopall: # stop immediately while True: # all productions exhausted? try: prod = prods[-1].nextProd(token=None) except Done, e: # ok prod = None except Missing, e: prod = None # last was a S operator which may End a Sequence, then ok if hasattr(lastprod, 'mayEnd') and not lastprod.mayEnd: wellformed = False self._log.error(u'%s: %s' % (name, e)) except ParseError, e: prod = None wellformed = False self._log.error(u'%s: %s' % (name, e)) else: if prods[-1].optional: prod = None elif prod and prod.optional: # ignore optional continue if prod and not prod.optional: wellformed = False self._log.error(u'%s: Missing token for production %r' % (name, str(prod))) break elif len(prods) > 1: # nested exhausted, next in parent prods.pop() else: break if not emptyOk and not len(seq): self._log.error(u'No content to parse.') return False, [], None, None # trim S from end seq.rstrip() return wellformed, seq, store, tokens class PreDef(object): """Predefined Prod definition for use in productions definition for ProdParser instances. """ types = cssutils.cssproductions.CSSProductions reHexcolor = re.compile(r'^\#(?:[0-9abcdefABCDEF]{3}|[0-9abcdefABCDEF]{6})$') @staticmethod def calc(toSeq=None, nextSor=False): return Prod(name=u'calcfunction', match=lambda t, v: u'calc(' == cssutils.helper.normalize(v), toSeq=toSeq, nextSor=nextSor) @staticmethod def char(name='char', char=u',', toSeq=None, stop=False, stopAndKeep=False, mayEnd=False, stopIfNoMoreMatch=False, optional=False, # WAS: optional=True, nextSor=False): "any CHAR" return Prod(name=name, match=lambda t, v: v == char, toSeq=toSeq, stop=stop, stopAndKeep=stopAndKeep, mayEnd=mayEnd, stopIfNoMoreMatch=stopIfNoMoreMatch, optional=optional, nextSor=nextSor) @staticmethod def comma(optional=False, toSeq=None): return PreDef.char(u'comma', u',', optional=optional, toSeq=toSeq) @staticmethod def comment(parent=None): return Prod(name=u'comment', match=lambda t, v: t == 'COMMENT', toSeq=lambda t, tokens: (t[0], cssutils.css.CSSComment([1], parentRule=parent)), optional=True ) @staticmethod def dimension(nextSor=False, stop=False): return Prod(name=u'dimension', match=lambda t, v: t == PreDef.types.DIMENSION, toSeq=lambda t, tokens: (t[0], cssutils.helper.normalize(t[1])), stop=stop, nextSor=nextSor) @staticmethod def function(toSeq=None, nextSor=False, toStore=None): return Prod(name=u'function', match=lambda t, v: t == PreDef.types.FUNCTION, toStore=toStore, toSeq=toSeq, nextSor=nextSor) @staticmethod def funcEnd(stop=False, mayEnd=False): ")" return PreDef.char(u'end FUNC ")"', u')', stop=stop, mayEnd=mayEnd) @staticmethod def hexcolor(stop=False, nextSor=False): "#123 or #123456" return Prod(name='HEX color', match=lambda t, v: ( t == PreDef.types.HASH and PreDef.reHexcolor.match(v) ), stop=stop, nextSor=nextSor) @staticmethod def ident(stop=False, toStore=None, nextSor=False): return Prod(name=u'ident', match=lambda t, v: t == PreDef.types.IDENT, stop=stop, toStore=toStore, nextSor=nextSor) @staticmethod def number(stop=False, toSeq=None, nextSor=False): return Prod(name=u'number', match=lambda t, v: t == PreDef.types.NUMBER, stop=stop, toSeq=toSeq, nextSor=nextSor) @staticmethod def percentage(stop=False, toSeq=None, nextSor=False): return Prod(name=u'percentage', match=lambda t, v: t == PreDef.types.PERCENTAGE, stop=stop, toSeq=toSeq, nextSor=nextSor) @staticmethod def string(stop=False, nextSor=False): "string delimiters are removed by default" return Prod(name=u'string', match=lambda t, v: t == PreDef.types.STRING, toSeq=lambda t, tokens: (t[0], cssutils.helper.stringvalue(t[1])), stop=stop, nextSor=nextSor) @staticmethod def S(name=u'whitespace', toSeq=None, optional=False): return Prod(name=name, match=lambda t, v: t == PreDef.types.S, toSeq=toSeq, optional=optional, mayEnd=True) @staticmethod def unary(stop=False, toSeq=None, nextSor=False): "+ or -" return Prod(name=u'unary +-', match=lambda t, v: v in (u'+', u'-'), optional=True, stop=stop, toSeq=toSeq, nextSor=nextSor) @staticmethod def uri(stop=False, nextSor=False): "'url(' and ')' are removed and URI is stripped" return Prod(name=u'URI', match=lambda t, v: t == PreDef.types.URI, toSeq=lambda t, tokens: (t[0], cssutils.helper.urivalue(t[1])), stop=stop, nextSor=nextSor) @staticmethod def unicode_range(stop=False, nextSor=False): "u+123456-abc normalized to lower `u`" return Prod(name='unicode-range', match=lambda t, v: t == PreDef.types.UNICODE_RANGE, toSeq=lambda t, tokens: (t[0], t[1].lower()), stop=stop, nextSor=nextSor ) @staticmethod def variable(toSeq=None, stop=False, nextSor=False, toStore=None): return Prod(name=u'variable', match=lambda t, v: u'var(' == cssutils.helper.normalize(v), toSeq=toSeq, toStore=toStore, stop=stop, nextSor=nextSor) # used for MarginRule for now: @staticmethod def unknownrule(name=u'@', toStore=None): """@rule dummy (matches ATKEYWORD to remove unknown rule tokens from stream:: @x; @x {...} no nested yet! """ def rule(tokens): saved = [] for t in tokens: saved.append(t) if (t[1] == u'}' or t[1] == u';'): return cssutils.css.CSSUnknownRule(saved) return Prod(name=name, match=lambda t, v: t == u'ATKEYWORD', toSeq=lambda t, tokens: (u'CSSUnknownRule', rule(pushtoken(t, tokens)) ), toStore=toStore ) cssutils-1.0/src/cssutils/_codec2.py0000666000175000017500000005627412126054542016244 0ustar hugohugo#!/usr/bin/env python """Python codec for CSS.""" __docformat__ = 'restructuredtext' __author__ = 'Walter Doerwald' __version__ = '$Id: util.py 1114 2008-03-05 13:22:59Z cthedot $' import codecs import marshal # We're using bits to store all possible candidate encodings (or variants, i.e. # we have two bits for the variants of UTF-16 and two for the # variants of UTF-32). # # Prefixes for various CSS encodings # UTF-8-SIG xEF xBB xBF # UTF-16 (LE) xFF xFE ~x00|~x00 # UTF-16 (BE) xFE xFF # UTF-16-LE @ x00 @ x00 # UTF-16-BE x00 @ # UTF-32 (LE) xFF xFE x00 x00 # UTF-32 (BE) x00 x00 xFE xFF # UTF-32-LE @ x00 x00 x00 # UTF-32-BE x00 x00 x00 @ # CHARSET @ c h a ... def detectencoding_str(input, final=False): """ Detect the encoding of the byte string ``input``, which contains the beginning of a CSS file. This function returns the detected encoding (or ``None`` if it hasn't got enough data), and a flag that indicates whether that encoding has been detected explicitely or implicitely. To detect the encoding the first few bytes are used (or if ``input`` is ASCII compatible and starts with a charset rule the encoding name from the rule). "Explicit" detection means that the bytes start with a BOM or a charset rule. If the encoding can't be detected yet, ``None`` is returned as the encoding. ``final`` specifies whether more data will be available in later calls or not. If ``final`` is true, ``detectencoding_str()`` will never return ``None`` as the encoding. """ # A bit for every candidate CANDIDATE_UTF_8_SIG = 1 CANDIDATE_UTF_16_AS_LE = 2 CANDIDATE_UTF_16_AS_BE = 4 CANDIDATE_UTF_16_LE = 8 CANDIDATE_UTF_16_BE = 16 CANDIDATE_UTF_32_AS_LE = 32 CANDIDATE_UTF_32_AS_BE = 64 CANDIDATE_UTF_32_LE = 128 CANDIDATE_UTF_32_BE = 256 CANDIDATE_CHARSET = 512 candidates = 1023 # all candidates li = len(input) if li>=1: # Check first byte c = input[0] if c != "\xef": candidates &= ~CANDIDATE_UTF_8_SIG if c != "\xff": candidates &= ~(CANDIDATE_UTF_32_AS_LE|CANDIDATE_UTF_16_AS_LE) if c != "\xfe": candidates &= ~CANDIDATE_UTF_16_AS_BE if c != "@": candidates &= ~(CANDIDATE_UTF_32_LE|CANDIDATE_UTF_16_LE|CANDIDATE_CHARSET) if c != "\x00": candidates &= ~(CANDIDATE_UTF_32_AS_BE|CANDIDATE_UTF_32_BE|CANDIDATE_UTF_16_BE) if li>=2: # Check second byte c = input[1] if c != "\xbb": candidates &= ~CANDIDATE_UTF_8_SIG if c != "\xfe": candidates &= ~(CANDIDATE_UTF_16_AS_LE|CANDIDATE_UTF_32_AS_LE) if c != "\xff": candidates &= ~CANDIDATE_UTF_16_AS_BE if c != "\x00": candidates &= ~(CANDIDATE_UTF_16_LE|CANDIDATE_UTF_32_AS_BE|CANDIDATE_UTF_32_LE|CANDIDATE_UTF_32_BE) if c != "@": candidates &= ~CANDIDATE_UTF_16_BE if c != "c": candidates &= ~CANDIDATE_CHARSET if li>=3: # Check third byte c = input[2] if c != "\xbf": candidates &= ~CANDIDATE_UTF_8_SIG if c != "c": candidates &= ~CANDIDATE_UTF_16_LE if c != "\x00": candidates &= ~(CANDIDATE_UTF_32_AS_LE|CANDIDATE_UTF_32_LE|CANDIDATE_UTF_32_BE) if c != "\xfe": candidates &= ~CANDIDATE_UTF_32_AS_BE if c != "h": candidates &= ~CANDIDATE_CHARSET if li>=4: # Check fourth byte c = input[3] if input[2:4] == "\x00\x00": candidates &= ~CANDIDATE_UTF_16_AS_LE if c != "\x00": candidates &= ~(CANDIDATE_UTF_16_LE|CANDIDATE_UTF_32_AS_LE|CANDIDATE_UTF_32_LE) if c != "\xff": candidates &= ~CANDIDATE_UTF_32_AS_BE if c != "@": candidates &= ~CANDIDATE_UTF_32_BE if c != "a": candidates &= ~CANDIDATE_CHARSET if candidates == 0: return ("utf-8", False) if not (candidates & (candidates-1)): # only one candidate remaining if candidates == CANDIDATE_UTF_8_SIG and li >= 3: return ("utf-8-sig", True) elif candidates == CANDIDATE_UTF_16_AS_LE and li >= 2: return ("utf-16", True) elif candidates == CANDIDATE_UTF_16_AS_BE and li >= 2: return ("utf-16", True) elif candidates == CANDIDATE_UTF_16_LE and li >= 4: return ("utf-16-le", False) elif candidates == CANDIDATE_UTF_16_BE and li >= 2: return ("utf-16-be", False) elif candidates == CANDIDATE_UTF_32_AS_LE and li >= 4: return ("utf-32", True) elif candidates == CANDIDATE_UTF_32_AS_BE and li >= 4: return ("utf-32", True) elif candidates == CANDIDATE_UTF_32_LE and li >= 4: return ("utf-32-le", False) elif candidates == CANDIDATE_UTF_32_BE and li >= 4: return ("utf-32-be", False) elif candidates == CANDIDATE_CHARSET and li >= 4: prefix = '@charset "' if input[:len(prefix)] == prefix: pos = input.find('"', len(prefix)) if pos >= 0: return (input[len(prefix):pos], True) # if this is the last call, and we haven't determined an encoding yet, # we default to UTF-8 if final: return ("utf-8", False) return (None, False) # dont' know yet def detectencoding_unicode(input, final=False): """ Detect the encoding of the unicode string ``input``, which contains the beginning of a CSS file. The encoding is detected from the charset rule at the beginning of ``input``. If there is no charset rule, ``"utf-8"`` will be returned. If the encoding can't be detected yet, ``None`` is returned. ``final`` specifies whether more data will be available in later calls or not. If ``final`` is true, ``detectencoding_unicode()`` will never return ``None``. """ prefix = u'@charset "' if input.startswith(prefix): pos = input.find(u'"', len(prefix)) if pos >= 0: return (input[len(prefix):pos], True) elif final or not prefix.startswith(input): # if this is the last call, and we haven't determined an encoding yet, # (or the string definitely doesn't start with prefix) we default to UTF-8 return ("utf-8", False) return (None, False) # don't know yet def _fixencoding(input, encoding, final=False): """ Replace the name of the encoding in the charset rule at the beginning of ``input`` with ``encoding``. If ``input`` doesn't starts with a charset rule, ``input`` will be returned unmodified. If the encoding can't be found yet, ``None`` is returned. ``final`` specifies whether more data will be available in later calls or not. If ``final`` is true, ``_fixencoding()`` will never return ``None``. """ prefix = u'@charset "' if len(input) > len(prefix): if input.startswith(prefix): pos = input.find(u'"', len(prefix)) if pos >= 0: if encoding.replace("_", "-").lower() == "utf-8-sig": encoding = u"utf-8" return prefix + encoding + input[pos:] # we haven't seen the end of the encoding name yet => fall through else: return input # doesn't start with prefix, so nothing to fix elif not prefix.startswith(input) or final: # can't turn out to be a @charset rule later (or there is no "later") return input if final: return input return None # don't know yet def decode(input, errors="strict", encoding=None, force=True): if encoding is None or not force: (_encoding, explicit) = detectencoding_str(input, True) if _encoding == "css": raise ValueError("css not allowed as encoding name") if (explicit and not force) or encoding is None: # Take the encoding from the input encoding = _encoding (input, consumed) = codecs.getdecoder(encoding)(input, errors) return (_fixencoding(input, unicode(encoding), True), consumed) def encode(input, errors="strict", encoding=None): consumed = len(input) if encoding is None: encoding = detectencoding_unicode(input, True)[0] if encoding.replace("_", "-").lower() == "utf-8-sig": input = _fixencoding(input, u"utf-8", True) else: input = _fixencoding(input, unicode(encoding), True) if encoding == "css": raise ValueError("css not allowed as encoding name") encoder = codecs.getencoder(encoding) return (encoder(input, errors)[0], consumed) def _bytes2int(bytes): # Helper: convert an 8 bit string into an ``int``. i = 0 for byte in bytes: i = (i<<8) + ord(byte) return i def _int2bytes(i): # Helper: convert an ``int`` into an 8-bit string. v = [] while i: v.insert(0, chr(i&0xff)) i >>= 8 return "".join(v) if hasattr(codecs, "IncrementalDecoder"): class IncrementalDecoder(codecs.IncrementalDecoder): def __init__(self, errors="strict", encoding=None, force=True): self.decoder = None self.encoding = encoding self.force = force codecs.IncrementalDecoder.__init__(self, errors) # Store ``errors`` somewhere else, # because we have to hide it in a property self._errors = errors self.buffer = u"".encode() self.headerfixed = False def iterdecode(self, input): for part in input: result = self.decode(part, False) if result: yield result result = self.decode("", True) if result: yield result def decode(self, input, final=False): # We're doing basically the same as a ``BufferedIncrementalDecoder``, # but since the buffer is only relevant until the encoding has been # detected (in which case the buffer of the underlying codec might # kick in), we're implementing buffering ourselves to avoid some # overhead. if self.decoder is None: input = self.buffer + input # Do we have to detect the encoding from the input? if self.encoding is None or not self.force: (encoding, explicit) = detectencoding_str(input, final) if encoding is None: # no encoding determined yet self.buffer = input # retry the complete input on the next call return u"" # no encoding determined yet, so no output elif encoding == "css": raise ValueError("css not allowed as encoding name") if (explicit and not self.force) or self.encoding is None: # Take the encoding from the input self.encoding = encoding self.buffer = "" # drop buffer, as the decoder might keep its own decoder = codecs.getincrementaldecoder(self.encoding) self.decoder = decoder(self._errors) if self.headerfixed: return self.decoder.decode(input, final) # If we haven't fixed the header yet, # the content of ``self.buffer`` is a ``unicode`` object output = self.buffer + self.decoder.decode(input, final) encoding = self.encoding if encoding.replace("_", "-").lower() == "utf-8-sig": encoding = "utf-8" newoutput = _fixencoding(output, unicode(encoding), final) if newoutput is None: # retry fixing the @charset rule (but keep the decoded stuff) self.buffer = output return u"" self.headerfixed = True return newoutput def reset(self): codecs.IncrementalDecoder.reset(self) self.decoder = None self.buffer = u"".encode() self.headerfixed = False def _geterrors(self): return self._errors def _seterrors(self, errors): # Setting ``errors`` must be done on the real decoder too if self.decoder is not None: self.decoder.errors = errors self._errors = errors errors = property(_geterrors, _seterrors) def getstate(self): if self.decoder is not None: state = (self.encoding, self.buffer, self.headerfixed, True, self.decoder.getstate()) else: state = (self.encoding, self.buffer, self.headerfixed, False, None) return ("", _bytes2int(marshal.dumps(state))) def setstate(self, state): state = _int2bytes(marshal.loads(state[1])) # ignore buffered input self.encoding = state[0] self.buffer = state[1] self.headerfixed = state[2] if state[3] is not None: self.decoder = codecs.getincrementaldecoder(self.encoding)(self._errors) self.decoder.setstate(state[4]) else: self.decoder = None if hasattr(codecs, "IncrementalEncoder"): class IncrementalEncoder(codecs.IncrementalEncoder): def __init__(self, errors="strict", encoding=None): self.encoder = None self.encoding = encoding codecs.IncrementalEncoder.__init__(self, errors) # Store ``errors`` somewhere else, # because we have to hide it in a property self._errors = errors self.buffer = u"" def iterencode(self, input): for part in input: result = self.encode(part, False) if result: yield result result = self.encode(u"", True) if result: yield result def encode(self, input, final=False): if self.encoder is None: input = self.buffer + input if self.encoding is not None: # Replace encoding in the @charset rule with the specified one encoding = self.encoding if encoding.replace("_", "-").lower() == "utf-8-sig": encoding = "utf-8" newinput = _fixencoding(input, unicode(encoding), final) if newinput is None: # @charset rule incomplete => Retry next time self.buffer = input return "" input = newinput else: # Use encoding from the @charset declaration self.encoding = detectencoding_unicode(input, final)[0] if self.encoding is not None: if self.encoding == "css": raise ValueError("css not allowed as encoding name") info = codecs.lookup(self.encoding) encoding = self.encoding if self.encoding.replace("_", "-").lower() == "utf-8-sig": input = _fixencoding(input, u"utf-8", True) self.encoder = info.incrementalencoder(self._errors) self.buffer = u"" else: self.buffer = input return "" return self.encoder.encode(input, final) def reset(self): codecs.IncrementalEncoder.reset(self) self.encoder = None self.buffer = u"" def _geterrors(self): return self._errors def _seterrors(self, errors): # Setting ``errors ``must be done on the real encoder too if self.encoder is not None: self.encoder.errors = errors self._errors = errors errors = property(_geterrors, _seterrors) def getstate(self): if self.encoder is not None: state = (self.encoding, self.buffer, True, self.encoder.getstate()) else: state = (self.encoding, self.buffer, False, None) return _bytes2int(marshal.dumps(state)) def setstate(self, state): state = _int2bytes(marshal.loads(state)) self.encoding = state[0] self.buffer = state[1] if state[2] is not None: self.encoder = codecs.getincrementalencoder(self.encoding)(self._errors) self.encoder.setstate(state[4]) else: self.encoder = None class StreamWriter(codecs.StreamWriter): def __init__(self, stream, errors="strict", encoding=None, header=False): codecs.StreamWriter.__init__(self, stream, errors) self.streamwriter = None self.encoding = encoding self._errors = errors self.buffer = u"" def encode(self, input, errors='strict'): li = len(input) if self.streamwriter is None: input = self.buffer + input li = len(input) if self.encoding is not None: # Replace encoding in the @charset rule with the specified one encoding = self.encoding if encoding.replace("_", "-").lower() == "utf-8-sig": encoding = "utf-8" newinput = _fixencoding(input, unicode(encoding), False) if newinput is None: # @charset rule incomplete => Retry next time self.buffer = input return ("", 0) input = newinput else: # Use encoding from the @charset declaration self.encoding = detectencoding_unicode(input, False)[0] if self.encoding is not None: if self.encoding == "css": raise ValueError("css not allowed as encoding name") self.streamwriter = codecs.getwriter(self.encoding)(self.stream, self._errors) encoding = self.encoding if self.encoding.replace("_", "-").lower() == "utf-8-sig": input = _fixencoding(input, u"utf-8", True) self.buffer = u"" else: self.buffer = input return ("", 0) return (self.streamwriter.encode(input, errors)[0], li) def _geterrors(self): return self._errors def _seterrors(self, errors): # Setting ``errors`` must be done on the streamwriter too if self.streamwriter is not None: self.streamwriter.errors = errors self._errors = errors errors = property(_geterrors, _seterrors) class StreamReader(codecs.StreamReader): def __init__(self, stream, errors="strict", encoding=None, force=True): codecs.StreamReader.__init__(self, stream, errors) self.streamreader = None self.encoding = encoding self.force = force self._errors = errors def decode(self, input, errors='strict'): if self.streamreader is None: if self.encoding is None or not self.force: (encoding, explicit) = detectencoding_str(input, False) if encoding is None: # no encoding determined yet return (u"", 0) # no encoding determined yet, so no output elif encoding == "css": raise ValueError("css not allowed as encoding name") if (explicit and not self.force) or self.encoding is None: # Take the encoding from the input self.encoding = encoding streamreader = codecs.getreader(self.encoding) streamreader = streamreader(self.stream, self._errors) (output, consumed) = streamreader.decode(input, errors) encoding = self.encoding if encoding.replace("_", "-").lower() == "utf-8-sig": encoding = "utf-8" newoutput = _fixencoding(output, unicode(encoding), False) if newoutput is not None: self.streamreader = streamreader return (newoutput, consumed) return (u"", 0) # we will create a new streamreader on the next call return self.streamreader.decode(input, errors) def _geterrors(self): return self._errors def _seterrors(self, errors): # Setting ``errors`` must be done on the streamreader too if self.streamreader is not None: self.streamreader.errors = errors self._errors = errors errors = property(_geterrors, _seterrors) if hasattr(codecs, "CodecInfo"): # We're running on Python 2.5 or better def search_function(name): if name == "css": return codecs.CodecInfo( name="css", encode=encode, decode=decode, incrementalencoder=IncrementalEncoder, incrementaldecoder=IncrementalDecoder, streamwriter=StreamWriter, streamreader=StreamReader, ) else: # If we're running on Python 2.4, define the utf-8-sig codec here def utf8sig_encode(input, errors='strict'): return (codecs.BOM_UTF8 + codecs.utf_8_encode(input, errors)[0], len(input)) def utf8sig_decode(input, errors='strict'): prefix = 0 if input[:3] == codecs.BOM_UTF8: input = input[3:] prefix = 3 (output, consumed) = codecs.utf_8_decode(input, errors, True) return (output, consumed+prefix) class UTF8SigStreamWriter(codecs.StreamWriter): def reset(self): codecs.StreamWriter.reset(self) try: del self.encode except AttributeError: pass def encode(self, input, errors='strict'): self.encode = codecs.utf_8_encode return utf8sig_encode(input, errors) class UTF8SigStreamReader(codecs.StreamReader): def reset(self): codecs.StreamReader.reset(self) try: del self.decode except AttributeError: pass def decode(self, input, errors='strict'): if len(input) < 3 and codecs.BOM_UTF8.startswith(input): # not enough data to decide if this is a BOM # => try again on the next call return (u"", 0) self.decode = codecs.utf_8_decode return utf8sig_decode(input, errors) def search_function(name): import encodings name = encodings.normalize_encoding(name) if name == "css": return (encode, decode, StreamReader, StreamWriter) elif name == "utf_8_sig": return (utf8sig_encode, utf8sig_decode, UTF8SigStreamReader, UTF8SigStreamWriter) codecs.register(search_function) # Error handler for CSS escaping def cssescape(exc): if not isinstance(exc, UnicodeEncodeError): raise TypeError("don't know how to handle %r" % exc) return (u"".join(u"\\%06x" % ord(c) for c in exc.object[exc.start:exc.end]), exc.end) codecs.register_error("cssescape", cssescape) cssutils-1.0/src/cssutils/__init__.py0000666000175000017500000003523112253304274016474 0ustar hugohugo#!/usr/bin/env python """cssutils - CSS Cascading Style Sheets library for Python Copyright (C) 2004-2013 Christof Hoeke cssutils is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program. If not, see . A Python package to parse and build CSS Cascading Style Sheets. DOM only, not any rendering facilities! Based upon and partly implementing the following specifications : `CSS 2.1 `__ General CSS rules and properties are defined here `CSS 2.1 Errata `__ A few errata, mainly the definition of CHARSET_SYM tokens `CSS3 Module: Syntax `__ Used in parts since cssutils 0.9.4. cssutils tries to use the features from CSS 2.1 and CSS 3 with preference to CSS3 but as this is not final yet some parts are from CSS 2.1 `MediaQueries `__ MediaQueries are part of ``stylesheets.MediaList`` since v0.9.4, used in @import and @media rules. `Namespaces `__ Added in v0.9.1, updated to definition in CSSOM in v0.9.4, updated in 0.9.5 for dev version `CSS3 Module: Pages Media `__ Most properties of this spec are implemented including MarginRules `Selectors `__ The selector syntax defined here (and not in CSS 2.1) should be parsable with cssutils (*should* mind though ;) ) `DOM Level 2 Style CSS `__ DOM for package css. 0.9.8 removes support for CSSValue and related API, see PropertyValue and Value API for now `DOM Level 2 Style Stylesheets `__ DOM for package stylesheets `CSSOM `__ A few details (mainly the NamespaceRule DOM) is taken from here. Plan is to move implementation to the stuff defined here which is newer but still no REC so might change anytime... The cssutils tokenizer is a customized implementation of `CSS3 Module: Syntax (W3C Working Draft 13 August 2003) `__ which itself is based on the CSS 2.1 tokenizer. It tries to be as compliant as possible but uses some (helpful) parts of the CSS 2.1 tokenizer. I guess cssutils is neither CSS 2.1 nor CSS 3 compliant but tries to at least be able to parse both grammars including some more real world cases (some CSS hacks are actually parsed and serialized). Both official grammars are not final nor bugfree but still feasible. cssutils aim is not to be fully compliant to any CSS specification (the specifications seem to be in a constant flow anyway) but cssutils *should* be able to read and write as many as possible CSS stylesheets "in the wild" while at the same time implement the official APIs which are well documented. Some minor extensions are provided as well. Please visit http://cthedot.de/cssutils/ for more details. Tested with Python 2.7.6 and 3.3.3 on Windows 8.1 64bit. This library may be used ``from cssutils import *`` which import subpackages ``css`` and ``stylesheets``, CSSParser and CSSSerializer classes only. Usage may be:: >>> from cssutils import * >>> parser = CSSParser() >>> sheet = parser.parseString(u'a { color: red}') >>> print sheet.cssText a { color: red } """ __all__ = ['css', 'stylesheets', 'CSSParser', 'CSSSerializer'] __docformat__ = 'restructuredtext' __author__ = 'Christof Hoeke with contributions by Walter Doerwald' __date__ = '$LastChangedDate:: $:' VERSION = '1.0' __version__ = '%s $Id$' % VERSION import sys if sys.version_info < (2,6): bytes = str import codec import os.path import urllib import urlparse import xml.dom # order of imports is important (partly circular) from . import util import errorhandler log = errorhandler.ErrorHandler() import css import stylesheets from parse import CSSParser from serialize import CSSSerializer ser = CSSSerializer() from profiles import Profiles profile = Profiles(log=log) # used by Selector defining namespace prefix '*' _ANYNS = -1 class DOMImplementationCSS(object): """This interface allows the DOM user to create a CSSStyleSheet outside the context of a document. There is no way to associate the new CSSStyleSheet with a document in DOM Level 2. This class is its *own factory*, as it is given to xml.dom.registerDOMImplementation which simply calls it and receives an instance of this class then. """ _features = [ ('css', '1.0'), ('css', '2.0'), ('stylesheets', '1.0'), ('stylesheets', '2.0') ] def createCSSStyleSheet(self, title, media): """ Creates a new CSSStyleSheet. title of type DOMString The advisory title. See also the Style Sheet Interfaces section. media of type DOMString The comma-separated list of media associated with the new style sheet. See also the Style Sheet Interfaces section. returns CSSStyleSheet: A new CSS style sheet. TODO: DOMException SYNTAX_ERR: Raised if the specified media string value has a syntax error and is unparsable. """ return css.CSSStyleSheet(title=title, media=media) def createDocument(self, *args): # not needed to HTML, also not for CSS? raise NotImplementedError def createDocumentType(self, *args): # not needed to HTML, also not for CSS? raise NotImplementedError def hasFeature(self, feature, version): return (feature.lower(), unicode(version)) in self._features xml.dom.registerDOMImplementation('cssutils', DOMImplementationCSS) def parseString(*a, **k): return CSSParser().parseString(*a, **k) parseString.__doc__ = CSSParser.parseString.__doc__ def parseFile(*a, **k): return CSSParser().parseFile(*a, **k) parseFile.__doc__ = CSSParser.parseFile.__doc__ def parseUrl(*a, **k): return CSSParser().parseUrl(*a, **k) parseUrl.__doc__ = CSSParser.parseUrl.__doc__ def parseStyle(*a, **k): return CSSParser().parseStyle(*a, **k) parseStyle.__doc__ = CSSParser.parseStyle.__doc__ # set "ser", default serializer def setSerializer(serializer): """Set the global serializer used by all class in cssutils.""" global ser ser = serializer def getUrls(sheet): """Retrieve all ``url(urlstring)`` values (in e.g. :class:`cssutils.css.CSSImportRule` or :class:`cssutils.css.CSSValue` objects of given `sheet`. :param sheet: :class:`cssutils.css.CSSStyleSheet` object whose URLs are yielded This function is a generator. The generated URL values exclude ``url(`` and ``)`` and surrounding single or double quotes. """ for importrule in (r for r in sheet if r.type == r.IMPORT_RULE): yield importrule.href def styleDeclarations(base): "recursive generator to find all CSSStyleDeclarations" if hasattr(base, 'cssRules'): for rule in base.cssRules: for s in styleDeclarations(rule): yield s elif hasattr(base, 'style'): yield base.style for style in styleDeclarations(sheet): for p in style.getProperties(all=True): for v in p.propertyValue: if v.type == 'URI': yield v.uri def replaceUrls(sheetOrStyle, replacer, ignoreImportRules=False): """Replace all URLs in :class:`cssutils.css.CSSImportRule` or :class:`cssutils.css.CSSValue` objects of given `sheetOrStyle`. :param sheetOrStyle: a :class:`cssutils.css.CSSStyleSheet` or a :class:`cssutils.css.CSSStyleDeclaration` which is changed in place :param replacer: a function which is called with a single argument `url` which is the current value of each url() excluding ``url(``, ``)`` and surrounding (single or double) quotes. :param ignoreImportRules: if ``True`` does not call `replacer` with URLs from @import rules. """ if not ignoreImportRules and not isinstance(sheetOrStyle, css.CSSStyleDeclaration): for importrule in (r for r in sheetOrStyle if r.type == r.IMPORT_RULE): importrule.href = replacer(importrule.href) def styleDeclarations(base): "recursive generator to find all CSSStyleDeclarations" if hasattr(base, 'cssRules'): for rule in base.cssRules: for s in styleDeclarations(rule): yield s elif hasattr(base, 'style'): yield base.style elif isinstance(sheetOrStyle, css.CSSStyleDeclaration): # base is a style already yield base for style in styleDeclarations(sheetOrStyle): for p in style.getProperties(all=True): for v in p.propertyValue: if v.type == v.URI: v.uri = replacer(v.uri) def resolveImports(sheet, target=None): """Recurcively combine all rules in given `sheet` into a `target` sheet. @import rules which use media information are tried to be wrapped into @media rules so keeping the media information. This may not work in all instances (if e.g. an @import rule itself contains an @import rule with different media infos or if it contains rules which may not be used inside an @media block like @namespace rules.). In these cases the @import rule is kept as in the original sheet and a WARNING is issued. :param sheet: in this given :class:`cssutils.css.CSSStyleSheet` all import rules are resolved and added to a resulting *flat* sheet. :param target: A :class:`cssutils.css.CSSStyleSheet` object which will be the resulting *flat* sheet if given :returns: given `target` or a new :class:`cssutils.css.CSSStyleSheet` object """ if not target: target = css.CSSStyleSheet(href=sheet.href, media=sheet.media, title=sheet.title) def getReplacer(targetbase): "Return a replacer which uses base to return adjusted URLs" basesch, baseloc, basepath, basequery, basefrag = urlparse.urlsplit(targetbase) basepath, basepathfilename = os.path.split(basepath) def replacer(uri): scheme, location, path, query, fragment = urlparse.urlsplit(uri) if not scheme and not location and not path.startswith(u'/'): # relative path, filename = os.path.split(path) combined = os.path.normpath(os.path.join(basepath, path, filename)) return urllib.pathname2url(combined) else: # keep anything absolute return uri return replacer for rule in sheet.cssRules: if rule.type == rule.CHARSET_RULE: pass elif rule.type == rule.IMPORT_RULE: log.info(u'Processing @import %r' % rule.href, neverraise=True) if rule.hrefFound: # add all rules of @import to current sheet target.add(css.CSSComment(cssText=u'/* START @import "%s" */' % rule.href)) try: # nested imports importedSheet = resolveImports(rule.styleSheet) except xml.dom.HierarchyRequestErr, e: log.warn(u'@import: Cannot resolve target, keeping rule: %s' % e, neverraise=True) target.add(rule) else: # adjust relative URI references log.info(u'@import: Adjusting paths for %r' % rule.href, neverraise=True) replaceUrls(importedSheet, getReplacer(rule.href), ignoreImportRules=True) # might have to wrap rules in @media if media given if rule.media.mediaText == u'all': mediaproxy = None else: keepimport = False for r in importedSheet: # check if rules present which may not be # combined with media if r.type not in (r.COMMENT, r.STYLE_RULE, r.IMPORT_RULE): keepimport = True break if keepimport: log.warn(u'Cannot combine imported sheet with' u' given media as other rules then' u' comments or stylerules found %r,' u' keeping %r' % (r, rule.cssText), neverraise=True) target.add(rule) continue # wrap in @media if media is not `all` log.info(u'@import: Wrapping some rules in @media ' u' to keep media: %s' % rule.media.mediaText, neverraise=True) mediaproxy = css.CSSMediaRule(rule.media.mediaText) for r in importedSheet: if mediaproxy: mediaproxy.add(r) else: # add to top sheet directly but are difficult anyway target.add(r) if mediaproxy: target.add(mediaproxy) else: # keep @import as it is log.error(u'Cannot get referenced stylesheet %r, keeping rule' % rule.href, neverraise=True) target.add(rule) else: target.add(rule) return target if __name__ == '__main__': print __doc__ cssutils-1.0/src/cssutils/cssproductions.py0000666000175000017500000001152212247637164020026 0ustar hugohugo"""productions for cssutils based on a mix of CSS 2.1 and CSS 3 Syntax productions - http://www.w3.org/TR/css3-syntax - http://www.w3.org/TR/css3-syntax/#grammar0 open issues - numbers contain "-" if present - HASH: #aaa is, #000 is not anymore, CSS2.1: 'nmchar': r'[_a-z0-9-]|{nonascii}|{escape}', CSS3: 'nmchar': r'[_a-z-]|{nonascii}|{escape}', """ __all__ = ['CSSProductions', 'MACROS', 'PRODUCTIONS'] __docformat__ = 'restructuredtext' __version__ = '$Id$' # a complete list of css3 macros MACROS = { 'nonascii': r'[^\0-\177]', 'unicode': r'\\[0-9A-Fa-f]{1,6}(?:{nl}|{s})?', #'escape': r'{unicode}|\\[ -~\200-\777]', 'escape': r'{unicode}|\\[^\n\r\f0-9a-f]', 'nmstart': r'[_a-zA-Z]|{nonascii}|{escape}', 'nmchar': r'[-_a-zA-Z0-9]|{nonascii}|{escape}', 'string1': r'"([^\n\r\f\\"]|\\{nl}|{escape})*"', 'string2': r"'([^\n\r\f\\']|\\{nl}|{escape})*'", 'invalid1': r'\"([^\n\r\f\\"]|\\{nl}|{escape})*', 'invalid2': r"\'([^\n\r\f\\']|\\{nl}|{escape})*", 'comment': r'\/\*[^*]*\*+([^/][^*]*\*+)*\/', 'ident': r'[-]?{nmstart}{nmchar}*', 'name': r'{nmchar}+', # TODO??? 'num': r'[+-]?[0-9]*\.[0-9]+|[+-]?[0-9]+', #r'[-]?\d+|[-]?\d*\.\d+', 'string': r'{string1}|{string2}', # from CSS2.1 'invalid': r'{invalid1}|{invalid2}', 'url': r'[\x09\x21\x23-\x26\x28\x2a-\x7E]|{nonascii}|{escape}', 's': r'\t|\r|\n|\f|\x20', 'w': r'{s}*', 'nl': r'\n|\r\n|\r|\f', 'A': r'A|a|\\0{0,4}(?:41|61)(?:\r\n|[ \t\r\n\f])?', 'B': r'B|b|\\0{0,4}(?:42|62)(?:\r\n|[ \t\r\n\f])?', 'C': r'C|c|\\0{0,4}(?:43|63)(?:\r\n|[ \t\r\n\f])?', 'D': r'D|d|\\0{0,4}(?:44|64)(?:\r\n|[ \t\r\n\f])?', 'E': r'E|e|\\0{0,4}(?:45|65)(?:\r\n|[ \t\r\n\f])?', 'F': r'F|f|\\0{0,4}(?:46|66)(?:\r\n|[ \t\r\n\f])?', 'G': r'G|g|\\0{0,4}(?:47|67)(?:\r\n|[ \t\r\n\f])?|\\G|\\g', 'H': r'H|h|\\0{0,4}(?:48|68)(?:\r\n|[ \t\r\n\f])?|\\H|\\h', 'I': r'I|i|\\0{0,4}(?:49|69)(?:\r\n|[ \t\r\n\f])?|\\I|\\i', 'K': r'K|k|\\0{0,4}(?:4b|6b)(?:\r\n|[ \t\r\n\f])?|\\K|\\k', 'L': r'L|l|\\0{0,4}(?:4c|6c)(?:\r\n|[ \t\r\n\f])?|\\L|\\l', 'M': r'M|m|\\0{0,4}(?:4d|6d)(?:\r\n|[ \t\r\n\f])?|\\M|\\m', 'N': r'N|n|\\0{0,4}(?:4e|6e)(?:\r\n|[ \t\r\n\f])?|\\N|\\n', 'O': r'O|o|\\0{0,4}(?:4f|6f)(?:\r\n|[ \t\r\n\f])?|\\O|\\o', 'P': r'P|p|\\0{0,4}(?:50|70)(?:\r\n|[ \t\r\n\f])?|\\P|\\p', 'R': r'R|r|\\0{0,4}(?:52|72)(?:\r\n|[ \t\r\n\f])?|\\R|\\r', 'S': r'S|s|\\0{0,4}(?:53|73)(?:\r\n|[ \t\r\n\f])?|\\S|\\s', 'T': r'T|t|\\0{0,4}(?:54|74)(?:\r\n|[ \t\r\n\f])?|\\T|\\t', 'U': r'U|u|\\0{0,4}(?:55|75)(?:\r\n|[ \t\r\n\f])?|\\U|\\u', 'V': r'V|v|\\0{0,4}(?:56|76)(?:\r\n|[ \t\r\n\f])?|\\V|\\v', 'X': r'X|x|\\0{0,4}(?:58|78)(?:\r\n|[ \t\r\n\f])?|\\X|\\x', 'Z': r'Z|z|\\0{0,4}(?:5a|7a)(?:\r\n|[ \t\r\n\f])?|\\Z|\\z', } # The following productions are the complete list of tokens # used by cssutils, a mix of CSS3 and some CSS2.1 productions. # The productions are **ordered**: PRODUCTIONS = [ # UTF8_BOM or UTF8_BOM_SIG will only be checked at beginning of CSS ('BOM', '\xfe\xff|\xef\xbb\xbf'), ('S', r'{s}+'), # 1st in list of general productions ('URI', r'{U}{R}{L}\({w}({string}|{url}*){w}\)'), ('IDENT', r'{A}{N}{D}'), ('FUNCTION', r'{ident}\('), ('UNICODE-RANGE', r'{U}\+[0-9A-Fa-f?]{1,6}(\-[0-9A-Fa-f]{1,6})?'), ('IDENT', r'{ident}'), ('DIMENSION', r'{num}{ident}'), ('PERCENTAGE', r'{num}\%'), ('NUMBER', r'{num}'), ('HASH', r'\#{name}'), ('COMMENT', r'{comment}'), #r'\/\*[^*]*\*+([^/][^*]*\*+)*\/'), ('STRING', r'{string}'), ('INVALID', r'{invalid}'), # from CSS2.1 ('ATKEYWORD', r'@{ident}'), # other keywords are done in the tokenizer ('INCLUDES', '\~\='), ('DASHMATCH', r'\|\='), ('PREFIXMATCH', r'\^\='), ('SUFFIXMATCH', r'\$\='), ('SUBSTRINGMATCH', r'\*\='), ('CDO', r'\<\!\-\-'), ('CDC', r'\-\-\>'), ('CHAR', r'[^"\']') # MUST always be last # valid ony at start so not checked everytime #('CHARSET_SYM', r'@charset '), # from Errata includes ending space! # checked specially if fullsheet is parsed ] class CSSProductions(object): """ most attributes are set later """ EOF = True # removed from productions as they simply are ATKEYWORD until # tokenizing CHARSET_SYM = u'CHARSET_SYM' FONT_FACE_SYM = u'FONT_FACE_SYM' MEDIA_SYM = u'MEDIA_SYM' IMPORT_SYM = u'IMPORT_SYM' NAMESPACE_SYM = u'NAMESPACE_SYM' PAGE_SYM = u'PAGE_SYM' VARIABLES_SYM = u'VARIABLES_SYM' for i, t in enumerate(PRODUCTIONS): setattr(CSSProductions, t[0].replace('-', '_'), t[0]) # may be enabled by settings.set _DXImageTransform = (u'FUNCTION', ur'progid\:DXImageTransform\.Microsoft\..+\(' ) cssutils-1.0/src/cssutils/serialize.py0000666000175000017500000012124712253154214016724 0ustar hugohugo#!/usr/bin/env python # -*- coding: utf-8 -*- """cssutils serializer""" __all__ = ['CSSSerializer', 'Preferences'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.helper import normalize import codecs import cssutils import helper import re import xml.dom def _escapecss(e): """ Escapes characters not allowed in the current encoding the CSS way with a backslash followed by a uppercase hex code point E.g. the german umlaut 'ä' is escaped as \E4 """ s = e.object[e.start:e.end] return u''.join([ur'\%s ' % str(hex(ord(x)))[2:] # remove 0x from hex .upper() for x in s]), e.end codecs.register_error('escapecss', _escapecss) class Preferences(object): """Control output of CSSSerializer. defaultAtKeyword = True Should the literal @keyword from src CSS be used or the default form, e.g. if ``True``: ``@import`` else: ``@i\mport`` defaultPropertyName = True Should the normalized propertyname be used or the one given in the src file, e.g. if ``True``: ``color`` else: ``c\olor`` Only used if ``keepAllProperties==False``. defaultPropertyPriority = True Should the normalized or literal priority be used, e.g. ``!important`` or ``!Im\portant`` importHrefFormat = None Uses hreftype if ``None`` or format ``"URI"`` if ``'string'`` or format ``url(URI)`` if ``'uri'`` indent = 4 * ' ' Indentation of e.g Properties inside a CSSStyleDeclaration indentClosingBrace = True Defines if closing brace of block is indented to match indentation of the block (default) oder match indentation of selector. indentSpecificities = False (**EXPERIMENTAL**) Indent rules with subset of Selectors and higher Specitivity keepAllProperties = True If ``True`` all properties set in the original CSSStylesheet are kept meaning even properties set twice with the exact same same name are kept! keepComments = True If ``False`` removes all CSSComments keepEmptyRules = False defines if empty rules like e.g. ``a {}`` are kept in the resulting serialized sheet keepUnknownAtRules = True defines if unknown @rules like e.g. ``@three-dee {}`` are kept in the serialized sheet keepUsedNamespaceRulesOnly = False if True only namespace rules which are actually used are kept lineNumbers = False Only used if a complete CSSStyleSheet is serialized. lineSeparator = u'\\n' How to end a line. This may be set to e.g. u'' for serializing of CSSStyleDeclarations usable in HTML style attribute. listItemSpacer = u' ' string which is used in ``css.SelectorList``, ``css.CSSValue`` and ``stylesheets.MediaList`` after the comma normalizedVarNames = True defines if variable names should be serialized normalized (they are used as being normalized anyway) omitLastSemicolon = True If ``True`` omits ; after last property of CSSStyleDeclaration omitLeadingZero = False defines if values between -1 and 1 should omit the 0, like ``.5px`` paranthesisSpacer = u' ' string which is used before an opening paranthesis like in a ``css.CSSMediaRule`` or ``css.CSSStyleRule`` propertyNameSpacer = u' ' string which is used after a Property name colon resolveVariables = True if ``True`` all variable references are tried to resolved and all CSSVariablesRules are removed from the output. Any variable reference not resolvable is simply kept untouched. selectorCombinatorSpacer = u' ' string which is used before and after a Selector combinator like +, > or ~. CSSOM defines a single space for this which is also the default in cssutils. spacer = u' ' general spacer, used e.g. by CSSUnknownRule validOnly = False if True only valid (Properties) are output A Property is valid if it is a known Property with a valid value. """ def __init__(self, **initials): """Always use named instead of positional parameters.""" self.useDefaults() for key, value in initials.items(): if value: self.__setattr__(key, value) def __repr__(self): return u"cssutils.css.%s(%s)" % (self.__class__.__name__, u', '.join(['\n %s=%r' % (p, self.__getattribute__(p)) for p in self.__dict__] )) def __str__(self): return u" ~`` encloses in prefs.selectorCombinatorSpacer - some other vals add ``*spacer`` except ``space=False`` """ prefspace = self.ser.prefs.spacer if val or type_ in ('STRING', 'URI'): # PRE if 'COMMENT' == type_: if self.ser.prefs.keepComments: val = val.cssText else: return elif 'S' == type_ and not keepS: return elif 'S' == type_ and keepS: val = u' ' elif 'STRING' == type_: # may be empty but MUST not be None if val is None: return val = helper.string(val) if not prefspace: self._remove_last_if_S() elif 'URI' == type_: val = helper.uri(val) elif 'HASH' == type_: val = self.ser._hash(val) elif hasattr(val, 'cssText'): val = val.cssText elif hasattr(val, 'mediaText'): val = val.mediaText elif val in u'+>~,:{;)]/=}' and not alwaysS: self._remove_last_if_S() # elif type_ in ('Property', cssutils.css.CSSRule.UNKNOWN_RULE): # val = val.cssText # elif type_ in ('NUMBER', 'DIMENSION', 'PERCENTAGE') and val == u'0': # # remove sign + or - if value is zero # # TODO: only for lenghts! # if self.out and self.out[-1] in u'+-': # del self.out[-1] # APPEND if indent or (val == u'}' and self.ser.prefs.indentClosingBrace): self.out.append(self.ser._indentblock(val, self.ser._level+1)) else: if val.endswith(u' '): self._remove_last_if_S() self.out.append(val) # POST if alwaysS and val in u'-+*/': # calc, */ not really but do anyway self.out.append(u' ') elif val in u'+>~': # enclose selector combinator self.out.insert(-1, self.ser.prefs.selectorCombinatorSpacer) self.out.append(self.ser.prefs.selectorCombinatorSpacer) elif u')' == val and not keepS: # CHAR funcend # TODO: pref? self.out.append(u' ') elif u',' == val: # list self.out.append(self.ser.prefs.listItemSpacer) elif u':' == val: # prop self.out.append(self.ser.prefs.propertyNameSpacer) elif u'{' == val: # block start self.out.insert(-1, self.ser.prefs.paranthesisSpacer) self.out.append(self.ser.prefs.lineSeparator) elif u';' == val or 'styletext' == type_: # end or prop or block self.out.append(self.ser.prefs.lineSeparator) elif val not in u'}[]()/=' and space and type_ != 'FUNCTION': self.out.append(self.ser.prefs.spacer) if type_ != 'STRING' and not self.ser.prefs.spacer and \ self.out and not self.out[-1].endswith(u' '): self.out.append(u' ') def value(self, delim=u'', end=None, keepS=False): "returns all items joined by delim" if not keepS: self._remove_last_if_S() if end: self.out.append(end) return delim.join(self.out) class CSSSerializer(object): """Serialize a CSSStylesheet and its parts. To use your own serializing method the easiest is to subclass CSS Serializer and overwrite the methods you like to customize. """ def __init__(self, prefs=None): """ :param prefs: instance of Preferences """ if not prefs: prefs = Preferences() self.prefs = prefs self._level = 0 # current nesting level # TODO: self._selectors = [] # holds SelectorList self._selectorlevel = 0 # current specificity nesting level def _atkeyword(self, rule): "returns default or source atkeyword depending on prefs" if self.prefs.defaultAtKeyword: return rule.atkeyword # default else: return rule._keyword def _indentblock(self, text, level): """ indent a block like a CSSStyleDeclaration to the given level which may be higher than self._level (e.g. for CSSStyleDeclaration) """ if not self.prefs.lineSeparator: return text return self.prefs.lineSeparator.join( [u'%s%s' % (level * self.prefs.indent, line) for line in text.split(self.prefs.lineSeparator)] ) def _propertyname(self, property, actual): """ used by all styledeclarations to get the propertyname used dependent on prefs setting defaultPropertyName and keepAllProperties """ if self.prefs.defaultPropertyName and not self.prefs.keepAllProperties: return property.name else: return actual def _linenumnbers(self, text): if self.prefs.lineNumbers: pad = len(str(text.count(self.prefs.lineSeparator)+1)) out = [] for i, line in enumerate(text.split(self.prefs.lineSeparator)): out.append((u'%*i: %s') % (pad, i+1, line)) text = self.prefs.lineSeparator.join(out) return text def _hash(self, val, type_=None): """ Short form of hash, e.g. #123 instead of #112233 """ # TODO: add pref for this! if len(val) == 7 and val[1] == val[2] and\ val[3] == val[4] and\ val[5] == val[6]: return u'#%s%s%s' % (val[1], val[3], val[5]) else: return val def _valid(self, x): "checks items valid property and prefs.validOnly" return not self.prefs.validOnly or (self.prefs.validOnly and x.valid) def do_CSSStyleSheet(self, stylesheet): """serializes a complete CSSStyleSheet""" useduris = stylesheet._getUsedURIs() out = [] for rule in stylesheet.cssRules: if self.prefs.keepUsedNamespaceRulesOnly and\ rule.NAMESPACE_RULE == rule.type and\ rule.namespaceURI not in useduris and ( rule.prefix or None not in useduris): continue cssText = rule.cssText if cssText: out.append(cssText) text = self._linenumnbers(self.prefs.lineSeparator.join(out)) # get encoding of sheet, defaults to UTF-8 try: encoding = stylesheet.cssRules[0].encoding except (IndexError, AttributeError): encoding = u'UTF-8' # TODO: py3 return b str but tests use unicode? return text.encode(encoding, u'escapecss') def do_CSSComment(self, rule): """ serializes CSSComment which consists only of commentText """ if rule._cssText and self.prefs.keepComments: return rule._cssText else: return u'' def do_CSSCharsetRule(self, rule): """ serializes CSSCharsetRule encoding: string always @charset "encoding"; no comments or other things allowed! """ if rule.wellformed: return u'@charset %s;' % helper.string(rule.encoding) else: return u'' def do_CSSVariablesRule(self, rule): """ serializes CSSVariablesRule media TODO variables CSSStyleDeclaration + CSSComments """ variablesText = rule.variables.cssText if variablesText and rule.wellformed and not self.prefs.resolveVariables: out = Out(self) out.append(self._atkeyword(rule)) for item in rule.seq: # assume comments { out.append(item.value, item.type) out.append(u'{') out.append(u'%s%s}' % (variablesText, self.prefs.lineSeparator), indent=1) return out.value() else: return u'' def do_CSSFontFaceRule(self, rule): """ serializes CSSFontFaceRule style CSSStyleDeclaration + CSSComments """ styleText = self.do_css_CSSStyleDeclaration(rule.style) if styleText and rule.wellformed: out = Out(self) out.append(self._atkeyword(rule)) for item in rule.seq: # assume comments { out.append(item.value, item.type) out.append(u'{') out.append(u'%s%s}' % (styleText, self.prefs.lineSeparator), indent=1) return out.value() else: return u'' def do_CSSImportRule(self, rule): """ serializes CSSImportRule href string media optional cssutils.stylesheets.medialist.MediaList name optional string + CSSComments """ if rule.wellformed: out = Out(self) out.append(self._atkeyword(rule)) for item in rule.seq: type_, val = item.type, item.value if 'href' == type_: # "href" or url(href) if self.prefs.importHrefFormat == 'string' or ( self.prefs.importHrefFormat != 'uri' and rule.hreftype == 'string'): out.append(val, 'STRING') else: out.append(val, 'URI') elif 'media' == type_: # media mediaText = self.do_stylesheets_medialist(val) if mediaText and mediaText != u'all': out.append(mediaText) elif 'name' == type_: out.append(val, 'STRING') else: out.append(val, type_) return out.value(end=u';') else: return u'' def do_CSSNamespaceRule(self, rule): """ serializes CSSNamespaceRule uri string prefix string + CSSComments """ if rule.wellformed: out = Out(self) out.append(self._atkeyword(rule)) for item in rule.seq: type_, val = item.type, item.value if 'namespaceURI' == type_: out.append(val, 'STRING') else: out.append(val, type_) return out.value(end=u';') else: return u'' def do_CSSMediaRule(self, rule): """ serializes CSSMediaRule + CSSComments """ # TODO: use Out()? # mediaquery if not rule.media.wellformed: return u'' # @media out = [self._atkeyword(rule)] if not len(self.prefs.spacer): # for now always with space as only webkit supports @mediaall? out.append(u' ') else: out.append(self.prefs.spacer) # might be empty out.append(self.do_stylesheets_medialist(rule.media)) # name, seq contains content after name only (Comments) if rule.name: out.append(self.prefs.spacer) nameout = Out(self) nameout.append(helper.string(rule.name)) for item in rule.seq: nameout.append(item.value, item.type) out.append(nameout.value()) # { out.append(self.prefs.paranthesisSpacer) out.append(u'{') out.append(self.prefs.lineSeparator) # rules rulesout = [] for r in rule.cssRules: rtext = r.cssText if rtext: # indent each line of cssText rulesout.append(self._indentblock(rtext, self._level + 1)) rulesout.append(self.prefs.lineSeparator) if not self.prefs.keepEmptyRules and not u''.join(rulesout).strip(): return u'' out.extend(rulesout) # } out.append(u'%s}' % ((self._level + int(self.prefs.indentClosingBrace)) * self.prefs.indent)) return u''.join(out) def do_CSSPageRule(self, rule): """ serializes CSSPageRule selectorText string style CSSStyleDeclaration cssRules CSSRuleList of MarginRule objects + CSSComments """ # rules rules = u'' rulesout = [] for r in rule.cssRules: rtext = r.cssText if rtext: rulesout.append(rtext) rulesout.append(self.prefs.lineSeparator) rulesText = u''.join(rulesout)#.strip() # omit semicolon only if no MarginRules styleText = self.do_css_CSSStyleDeclaration(rule.style, omit=not rulesText) if (styleText or rulesText) and rule.wellformed: out = Out(self) out.append(self._atkeyword(rule)) out.append(rule.selectorText) out.append(u'{') if styleText: if not rulesText: out.append(u'%s%s' % (styleText, self.prefs.lineSeparator ), indent=1) else: out.append(styleText, type_='styletext', indent=1, space=False) if rulesText: out.append(rulesText, indent=1) #? self._level -= 1 out.append(u'}') self._level += 1 return out.value() else: return u'' def do_CSSPageRuleSelector(self, seq): "Serialize selector of a CSSPageRule" out = Out(self) for item in seq: if item.type == 'IDENT': out.append(item.value, item.type, space=False) else: out.append(item.value, item.type) return out.value() def do_MarginRule(self, rule): """ serializes MarginRule atkeyword string style CSSStyleDeclaration + CSSComments """ # might not be set at all?! if rule.atkeyword: styleText = self.do_css_CSSStyleDeclaration(rule.style) if styleText and rule.wellformed: out = Out(self) # # use seq but styledecl missing # for item in rule.seq: # if item.type == 'ATKEYWORD': # # move logic to Out # out.append(self._atkeyword(rule), type_=item.type) # else: # print type_, val # out.append(item.value, item.type) # return out.value() # ok for now: out.append(self._atkeyword(rule), type_='ATKEYWORD') out.append(u'{') out.append(u'%s%s' % (self._indentblock(styleText, self._level+1), self.prefs.lineSeparator)) out.append(u'}') return out.value() return u'' def do_CSSUnknownRule(self, rule): """ serializes CSSUnknownRule anything until ";" or "{...}" + CSSComments """ if rule.wellformed and self.prefs.keepUnknownAtRules: out = Out(self) out.append(rule.atkeyword) stacks = [] for item in rule.seq: type_, val = item.type, item.value # PRE if u'}' == val: # close last open item on stack stackblock = stacks.pop().value() if stackblock: val = self._indentblock( stackblock + self.prefs.lineSeparator + val, min(1, len(stacks)+1)) else: val = self._indentblock(val, min(1, len(stacks)+1)) # APPEND if stacks: stacks[-1].append(val, type_) else: out.append(val, type_) # POST if u'{' == val: # new stack level stacks.append(Out(self)) return out.value() else: return u'' def do_CSSStyleRule(self, rule): """ serializes CSSStyleRule selectorList style + CSSComments """ # TODO: use Out() # prepare for element nested rules # TODO: sort selectors! if self.prefs.indentSpecificities: # subselectorlist? elements = set([s.element for s in rule.selectorList]) specitivities = [s.specificity for s in rule.selectorList] for selector in self._selectors: lastelements = set([s.element for s in selector]) if elements.issubset(lastelements): # higher specificity? lastspecitivities = [s.specificity for s in selector] if specitivities > lastspecitivities: self._selectorlevel += 1 break elif self._selectorlevel > 0: self._selectorlevel -= 1 else: # save new reference self._selectors.append(rule.selectorList) self._selectorlevel = 0 # TODO ^ RESOLVE!!!! selectorText = self.do_css_SelectorList(rule.selectorList) if not selectorText or not rule.wellformed: return u'' self._level += 1 styleText = u'' try: styleText = self.do_css_CSSStyleDeclaration(rule.style) finally: self._level -= 1 if not styleText: if self.prefs.keepEmptyRules: return u'%s%s{}' % (selectorText, self.prefs.paranthesisSpacer) else: return self._indentblock( u'%s%s{%s%s%s%s}' % ( selectorText, self.prefs.paranthesisSpacer, self.prefs.lineSeparator, self._indentblock(styleText, self._level + 1), self.prefs.lineSeparator, (self._level + int(self.prefs.indentClosingBrace)) * self.prefs.indent), self._selectorlevel) def do_css_SelectorList(self, selectorlist): "comma-separated list of Selectors" # does not need Out() as it is too simple if selectorlist.wellformed: out = [] for part in selectorlist.seq: if isinstance(part, cssutils.css.Selector): out.append(part.selectorText) else: out.append(part) # should not happen sep = u',%s' % self.prefs.listItemSpacer return sep.join(out) else: return u'' def do_css_Selector(self, selector): """ a single Selector including comments an element has syntax (namespaceURI, name) where namespaceURI may be: - cssutils._ANYNS => ``*|name`` - None => ``name`` - u'' => ``|name`` - any other value: => ``prefix|name`` """ if selector.wellformed: out = Out(self) DEFAULTURI = selector._namespaces.get('', None) for item in selector.seq: type_, val = item.type, item.value if isinstance(val, tuple): # namespaceURI|name (element or attribute) namespaceURI, name = val if DEFAULTURI == namespaceURI or (not DEFAULTURI and namespaceURI is None): out.append(name, type_, space=False) else: if namespaceURI == cssutils._ANYNS: prefix = u'*' else: try: prefix = selector._namespaces.prefixForNamespaceURI( namespaceURI) except IndexError: prefix = u'' out.append(u'%s|%s' % (prefix, name), type_, space=False) else: out.append(val, type_, space=False, keepS=True) return out.value() else: return u'' def do_css_CSSVariablesDeclaration(self, variables): """Variables of CSSVariableRule.""" if len(variables.seq) > 0: out = Out(self) lastitem = len(variables.seq) - 1 for i, item in enumerate(variables.seq): type_, val = item.type, item.value if u'var' == type_: name, cssvalue = val if self.prefs.normalizedVarNames: name = normalize(name) out.append(name) out.append(u':') out.append(cssvalue.cssText) if i < lastitem or not self.prefs.omitLastSemicolon: out.append(u';') elif isinstance(val, cssutils.css.CSSComment): # CSSComment out.append(val, 'COMMENT') out.append(self.prefs.lineSeparator) else: out.append(val.cssText, type_) out.append(self.prefs.lineSeparator) return out.value().strip() else: return u'' def do_css_CSSStyleDeclaration(self, style, separator=None, omit=True): """ Style declaration of CSSStyleRule """ # TODO: use Out() # may be comments only if len(style.seq) > 0: if separator is None: separator = self.prefs.lineSeparator if self.prefs.keepAllProperties: # all seq = style.seq else: # only effective ones _effective = style.getProperties() seq = [item for item in style.seq if (isinstance(item.value, cssutils.css.Property) and item.value in _effective) or not isinstance(item.value, cssutils.css.Property)] out = [] omitLastSemicolon = omit and self.prefs.omitLastSemicolon for i, item in enumerate(seq): type_, val = item.type, item.value if isinstance(val, cssutils.css.CSSComment): # CSSComment if self.prefs.keepComments: out.append(val.cssText) out.append(separator) elif isinstance(val, cssutils.css.Property): # PropertySimilarNameList if val.cssText: out.append(val.cssText) if not (omitLastSemicolon and i==len(seq)-1): out.append(u';') out.append(separator) elif isinstance(val, cssutils.css.CSSUnknownRule): # @rule out.append(val.cssText) out.append(separator) else: # ? out.append(val) out.append(separator) if out and out[-1] == separator: del out[-1] return u''.join(out) else: return u'' def do_Property(self, property): """ Style declaration of CSSStyleRule Property has a seqs attribute which contains seq lists for name, a CSSvalue and a seq list for priority """ # TODO: use Out() out = [] if property.seqs[0] and property.wellformed and self._valid(property): nameseq, value, priorityseq = property.seqs #name for part in nameseq: if hasattr(part, 'cssText'): out.append(part.cssText) elif property.literalname == part: out.append(self._propertyname(property, part)) else: out.append(part) if out and (not property._mediaQuery or property._mediaQuery and value.cssText): # MediaQuery may consist of name only out.append(u':') out.append(self.prefs.propertyNameSpacer) # value out.append(value.cssText) # priority if out and priorityseq: out.append(u' ') for part in priorityseq: if hasattr(part, 'cssText'): # comments out.append(part.cssText) else: if part == property.literalpriority and\ self.prefs.defaultPropertyPriority: out.append(property.priority) else: out.append(part) return u''.join(out) def do_Property_priority(self, priorityseq): """ a Properties priority "!" S* "important" """ # TODO: use Out() out = [] for part in priorityseq: if hasattr(part, 'cssText'): # comments out.append(u' ') out.append(part.cssText) out.append(u' ') else: out.append(part) return u''.join(out).strip() def do_css_PropertyValue(self, value, valuesOnly=False): """Serializes a PropertyValue""" if not value: return u'' else: out = Out(self) for item in value.seq: type_, val = item.type, item.value if valuesOnly and type_ == cssutils.css.CSSComment: continue elif hasattr(val, 'cssText'): # RGBColor or CSSValue if a CSSValueList out.append(val.cssText, type_) else: if val and val[0] == val[-1] and val[0] in '\'"': val = helper.string(val[1:-1]) # S must be kept! in between values but no extra space out.append(val, type_) return out.value() def _strip_zeros(self, s): i = s.index(u'.') + 2 a, b = s[0:i], s[i:len(s)] b = b.rstrip('0') return a + b def do_css_Value(self, value, valuesOnly=None): """Serializes a Value, valuesOnly is ignored""" if not value: return u'' else: out = Out(self) if value.type in (u'DIMENSION', u'NUMBER', u'PERCENTAGE'): dim = value.dimension or u'' if value.value == 0: val = u'0' if value.dimension in ('cm', 'mm', 'in', 'px', 'pc', 'pt', 'em', 'ex'): dim = u'' elif value.value == int(value.value): # cut off after . which is zero anyway val = unicode(int(value.value)) elif self.prefs.omitLeadingZero and -1 < value.value < 1: v = self._strip_zeros(u'%f' % value.value) # issue #27 val = v if value._sign == u'-': val = v[0] + v[2:] else: val = v[1:] else: val = self._strip_zeros(u'%f' % value.value) # issue #27 # keep '+' if given if value.value != 0 and value._sign == u'+': sign = u'+' else: sign = u'' out.append(sign + val + dim, value.type) else: # e.g. URI out.append(value.value, value.type) return out.value() def do_css_ColorValue(self, value, valuesOnly=False): """Serialize a ColorValue, a HASH simple value or FUNCTION""" try: return {'FUNCTION': self.do_css_CSSFunction, 'HASH': self.do_css_Value, 'IDENT': self.do_css_Value }[value.colorType](value, valuesOnly=valuesOnly) except KeyError, e: return u'' def do_css_CSSFunction(self, cssvalue, valuesOnly=False): """Serialize a CSS function value""" if not cssvalue: return u'' else: out = Out(self) for item in cssvalue.seq: type_, val = item.type, item.value if valuesOnly and type_ == cssutils.css.CSSComment: continue out.append(val, type_) return out.value() def do_css_CSSCalc(self, cssvalue, valuesOnly=False): """Serialize a CSS calc value""" if not cssvalue: return u'' else: out = Out(self) for item in cssvalue.seq: type_, val = item.type, item.value if valuesOnly and type_ == cssutils.css.CSSComment: continue elif hasattr(val, 'cssText'): # RGBColor or CSSValue if a CSSValueList out.append(val.cssText, type_) elif type_ == 'CHAR' and val in u'-+*/': out.append(val, type_, alwaysS=True) else: out.append(val, type_) return out.value() def do_css_MSValue(self, cssvalue, valuesOnly=False): """Serialize an ExpressionValue (IE only), should at least keep the original syntax""" if not cssvalue: return u'' else: out = Out(self) for item in cssvalue.seq: type_, val = item.type, item.value #val = self._possiblezero(cssvalue, type_, val) # do no send type_ so no special cases! out.append(val, None, space=False) return out.value() def do_css_CSSVariable(self, variable, IGNORED=False): """Serializes a CSSVariable""" if not variable or not variable.name: return u'' else: out = Out(self) v = variable.value if self.prefs.resolveVariables and v: # resolve variable out.append(v) else: # keep var(NAME) out.append(u'var(', 'FUNCTION') out.append(variable.name, 'IDENT') if variable.fallback: out.append(u',', 'COMMA') out.append(variable.fallback.cssText) out.append(u')') return out.value() def do_stylesheets_medialist(self, medialist): """ comma-separated list of media, default is 'all' If "all" is in the list, every other media *except* "handheld" will be stripped. This is because how Opera handles CSS for PDAs. """ if len(medialist) == 0: return u'all' else: seq = medialist.seq out = Out(self) firstdone = False for item in seq: type_, val = item.type, item.value if type_ == 'MediaQuery': if firstdone: out.append(u',', 'CHAR') else: firstdone = True out.append(item.value, item.type) return out.value() def do_stylesheets_mediaquery(self, mediaquery): """ a single media used in medialist """ if not mediaquery.wellformed: return u'' else: withsemi = [] nextmq = False for item in mediaquery.seq: type_, val = item.type, item.value if type_ == 'MediaQuery' and nextmq: withsemi.append(('CHAR', u',')) nextmq = False else: nextmq = True withsemi.append((type_, val)) out = Out(self) for t, v, in withsemi: out.append(v, t) #for item in mediaquery.seq: # type_, val = item.type, item.value # out.append(val, type_)#, space=False) return out.value() #if mediaquery.wellformed: # out = [] # for part in mediaquery.seq: # if isinstance(part, cssutils.css.Property): # Property # out.append(u'(%s)' % part.cssText) # elif hasattr(part, 'cssText'): # comments # out.append(part.cssText) # else: # # TODO: media queries! # out.append(part) # return u' '.join(out) #else: # return u'' cssutils-1.0/src/cssutils/css/0000755000175000017500000000000012416502415015141 5ustar hugohugocssutils-1.0/src/cssutils/css/cssfontfacerule.py0000666000175000017500000001602712126054542020714 0ustar hugohugo"""CSSFontFaceRule implements DOM Level 2 CSS CSSFontFaceRule. From cssutils 0.9.6 additions from CSS Fonts Module Level 3 are added http://www.w3.org/TR/css3-fonts/. """ __all__ = ['CSSFontFaceRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssstyledeclaration import CSSStyleDeclaration import cssrule import cssutils import xml.dom class CSSFontFaceRule(cssrule.CSSRule): """ The CSSFontFaceRule interface represents a @font-face rule in a CSS style sheet. The @font-face rule is used to hold a set of font descriptions. Format:: font_face : FONT_FACE_SYM S* '{' S* declaration [ ';' S* declaration ]* '}' S* ; cssutils uses a :class:`~cssutils.css.CSSStyleDeclaration` to represent the font descriptions. For validation a specific profile is used though were some properties have other valid values than when used in e.g. a :class:`~cssutils.css.CSSStyleRule`. """ def __init__(self, style=None, parentRule=None, parentStyleSheet=None, readonly=False): """ If readonly allows setting of properties in constructor only. :param style: CSSStyleDeclaration used to hold any font descriptions for this CSSFontFaceRule """ super(CSSFontFaceRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = u'@font-face' if style: self.style = style else: self.style = CSSStyleDeclaration() self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(style=%r)" % ( self.__class__.__name__, self.style.cssText) def __str__(self): return u"" % ( self.__class__.__name__, self.style.cssText, self.valid, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSFontFaceRule(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(CSSFontFaceRule, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) attoken = self._nexttoken(tokenizer, None) if self._type(attoken) != self._prods.FONT_FACE_SYM: self._log.error(u'CSSFontFaceRule: No CSSFontFaceRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: newStyle = CSSStyleDeclaration(parentRule=self) ok = True beforetokens, brace = self._tokensupto2(tokenizer, blockstartonly=True, separateEnd=True) if self._tokenvalue(brace) != u'{': ok = False self._log.error(u'CSSFontFaceRule: No start { of style ' u'declaration found: %r' % self._valuestr(cssText), brace) # parse stuff before { which should be comments and S only new = {'wellformed': True} newseq = self._tempSeq() beforewellformed, expected = self._parse(expected=':', seq=newseq, tokenizer=self._tokenize2(beforetokens), productions={}) ok = ok and beforewellformed and new['wellformed'] styletokens, braceorEOFtoken = self._tokensupto2(tokenizer, blockendonly=True, separateEnd=True) val, type_ = self._tokenvalue(braceorEOFtoken),\ self._type(braceorEOFtoken) if val != u'}' and type_ != 'EOF': ok = False self._log.error(u'CSSFontFaceRule: No "}" after style ' u'declaration found: %r' % self._valuestr(cssText)) nonetoken = self._nexttoken(tokenizer) if nonetoken: ok = False self._log.error(u'CSSFontFaceRule: Trailing content found.', token=nonetoken) if 'EOF' == type_: # add again as style needs it styletokens.append(braceorEOFtoken) # SET, may raise: newStyle.cssText = styletokens if ok: # contains probably comments only (upto ``{``) self._setSeq(newseq) self.style = newStyle cssText = property(_getCssText, _setCssText, doc=u"(DOM) The parsable textual representation of this " u"rule.") def _setStyle(self, style): """ :param style: a CSSStyleDeclaration or string """ self._checkReadonly() if isinstance(style, basestring): self._style = CSSStyleDeclaration(cssText=style, parentRule=self) else: style._parentRule = self self._style = style style = property(lambda self: self._style, _setStyle, doc=u"(DOM) The declaration-block of this rule set, " u"a :class:`~cssutils.css.CSSStyleDeclaration`.") type = property(lambda self: self.FONT_FACE_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") def _getValid(self): needed = ['font-family', 'src'] for p in self.style.getProperties(all=True): if not p.valid: return False try: needed.remove(p.name) except ValueError: pass return not bool(needed) valid = property(_getValid, doc=u"CSSFontFace is valid if properties `font-family` " u"and `src` are set and all properties are valid.") # constant but needed: wellformed = property(lambda self: True) cssutils-1.0/src/cssutils/css/csscomment.py0000666000175000017500000000542212250172130017666 0ustar hugohugo"""CSSComment is not defined in DOM Level 2 at all but a cssutils defined class only. Implements CSSRule which is also extended for a CSSComment rule type. """ __all__ = ['CSSComment'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssrule import cssutils import xml.dom class CSSComment(cssrule.CSSRule): """ Represents a CSS comment (cssutils only). Format:: /*...*/ """ def __init__(self, cssText=None, parentRule=None, parentStyleSheet=None, readonly=False): super(CSSComment, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._cssText = None if cssText: self._setCssText(cssText) self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(cssText=%r)" % ( self.__class__.__name__, self.cssText) def __str__(self): return u"" % ( self.__class__.__name__, self.cssText, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSComment(self) def _setCssText(self, cssText): """ :param cssText: textual text to set or tokenlist which is not tokenized anymore. May also be a single token for this rule :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(CSSComment, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) commenttoken = self._nexttoken(tokenizer) unexpected = self._nexttoken(tokenizer) if not commenttoken or\ self._type(commenttoken) != self._prods.COMMENT or\ unexpected: self._log.error(u'CSSComment: Not a CSSComment: %r' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: self._cssText = self._tokenvalue(commenttoken) cssText = property(_getCssText, _setCssText, doc=u"The parsable textual representation of this rule.") type = property(lambda self: self.COMMENT, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") # constant but needed: wellformed = property(lambda self: True) cssutils-1.0/src/cssutils/css/cssmediarule.py0000666000175000017500000003074512250171474020213 0ustar hugohugo"""CSSMediaRule implements DOM Level 2 CSS CSSMediaRule.""" __all__ = ['CSSMediaRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssrule import cssutils import xml.dom class CSSMediaRule(cssrule.CSSRuleRules): """ Objects implementing the CSSMediaRule interface can be identified by the MEDIA_RULE constant. On these objects the type attribute must return the value of that constant. Format:: : MEDIA_SYM S* medium [ COMMA S* medium ]* STRING? # the name LBRACE S* ruleset* '}' S*; ``cssRules`` All Rules in this media rule, a :class:`~cssutils.css.CSSRuleList`. """ def __init__(self, mediaText='all', name=None, parentRule=None, parentStyleSheet=None, readonly=False): """constructor""" super(CSSMediaRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = u'@media' # 1. media if mediaText: self.media = mediaText else: self.media = cssutils.stylesheets.MediaList() self.name = name self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(mediaText=%r)" % ( self.__class__.__name__, self.media.mediaText) def __str__(self): return u"" % ( self.__class__.__name__, self.media.mediaText, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSMediaRule(self) def _setCssText(self, cssText): """ :param cssText: a parseable string or a tuple of (cssText, dict-of-namespaces) :Exceptions: - :exc:`~xml.dom.NamespaceErr`: Raised if a specified selector uses an unknown namespace prefix. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ # media "name"? { cssRules } super(CSSMediaRule, self)._setCssText(cssText) # might be (cssText, namespaces) cssText, namespaces = self._splitNamespacesOff(cssText) try: # use parent style sheet ones if available namespaces = self.parentStyleSheet.namespaces except AttributeError: pass tokenizer = self._tokenize2(cssText) attoken = self._nexttoken(tokenizer, None) if self._type(attoken) != self._prods.MEDIA_SYM: self._log.error(u'CSSMediaRule: No CSSMediaRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: # save if parse goes wrong oldMedia = self._media oldName = self._name oldCssRules = self._cssRules ok = True # media mediatokens, end = self._tokensupto2(tokenizer, mediaqueryendonly=True, separateEnd=True) if u'{' == self._tokenvalue(end)\ or self._prods.STRING == self._type(end): self.media = cssutils.stylesheets.MediaList(parentRule=self) # TODO: remove special case self.media.mediaText = mediatokens ok = ok and self.media.wellformed else: ok = False # name (optional) name = None nameseq = self._tempSeq() if self._prods.STRING == self._type(end): name = self._stringtokenvalue(end) # TODO: for now comments are lost after name nametokens, end = self._tokensupto2(tokenizer, blockstartonly=True, separateEnd=True) wellformed, expected = self._parse(None, nameseq, nametokens, {}) if not wellformed: ok = False self._log.error(u'CSSMediaRule: Syntax Error: %s' % self._valuestr(cssText)) # check for { if u'{' != self._tokenvalue(end): self._log.error(u'CSSMediaRule: No "{" found: %s' % self._valuestr(cssText)) return # cssRules cssrulestokens, braceOrEOF = self._tokensupto2(tokenizer, mediaendonly=True, separateEnd=True) nonetoken = self._nexttoken(tokenizer, None) if 'EOF' == self._type(braceOrEOF): # HACK!!! # TODO: Not complete, add EOF to rule and } to @media cssrulestokens.append(braceOrEOF) braceOrEOF = ('CHAR', '}', 0, 0) self._log.debug(u'CSSMediaRule: Incomplete, adding "}".', token=braceOrEOF, neverraise=True) if u'}' != self._tokenvalue(braceOrEOF): self._log.error(u'CSSMediaRule: No "}" found.', token=braceOrEOF) elif nonetoken: self._log.error(u'CSSMediaRule: Trailing content found.', token=nonetoken) else: # for closures: must be a mutable new = {'wellformed': True } def COMMENT(expected, seq, token, tokenizer=None): self.insertRule(cssutils.css.CSSComment([token], parentRule=self, parentStyleSheet=self.parentStyleSheet)) return expected def ruleset(expected, seq, token, tokenizer): rule = cssutils.css.CSSStyleRule(parentRule=self, parentStyleSheet=self.parentStyleSheet) rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: self.insertRule(rule) return expected def atrule(expected, seq, token, tokenizer): # TODO: get complete rule! tokens = self._tokensupto2(tokenizer, token) atval = self._tokenvalue(token) if atval in ('@charset ', '@font-face', '@import', '@namespace', '@page', '@media', '@variables'): self._log.error(u'CSSMediaRule: This rule is not ' u'allowed in CSSMediaRule - ignored: ' u'%s.' % self._valuestr(tokens), token = token, error=xml.dom.HierarchyRequestErr) else: rule = cssutils.css.CSSUnknownRule(tokens, parentRule=self, parentStyleSheet=self.parentStyleSheet) if rule.wellformed: self.insertRule(rule) return expected # save for possible reset oldCssRules = self.cssRules self.cssRules = cssutils.css.CSSRuleList() seq = [] # not used really tokenizer = iter(cssrulestokens) wellformed, expected = self._parse(braceOrEOF, seq, tokenizer, { 'COMMENT': COMMENT, 'CHARSET_SYM': atrule, 'FONT_FACE_SYM': atrule, 'IMPORT_SYM': atrule, 'NAMESPACE_SYM': atrule, 'PAGE_SYM': atrule, 'MEDIA_SYM': atrule, 'ATKEYWORD': atrule }, default=ruleset, new=new) ok = ok and wellformed if ok: self.name = name self._setSeq(nameseq) else: self._media = oldMedia self._cssRules = oldCssRules cssText = property(_getCssText, _setCssText, doc=u"(DOM) The parsable textual representation of this " u"rule.") def _setName(self, name): if isinstance(name, basestring) or name is None: # "" or '' if not name: name = None self._name = name else: self._log.error(u'CSSImportRule: Not a valid name: %s' % name) name = property(lambda self: self._name, _setName, doc=u"An optional name for this media rule.") def _setMedia(self, media): """ :param media: a :class:`~cssutils.stylesheets.MediaList` or string """ self._checkReadonly() if isinstance(media, basestring): self._media = cssutils.stylesheets.MediaList(mediaText=media, parentRule=self) else: media._parentRule = self self._media = media # NOT IN @media seq at all?! # # update seq # for i, item in enumerate(self.seq): # if item.type == 'media': # self._seq[i] = (self._media, 'media', None, None) # break # else: # # insert after @media if not in seq at all # self.seq.insert(0, # self._media, 'media', None, None) media = property(lambda self: self._media, _setMedia, doc=u"(DOM) A list of media types for this rule " u"of type :class:`~cssutils.stylesheets.MediaList`.") def insertRule(self, rule, index=None): """Implements base ``insertRule``.""" rule, index = self._prepareInsertRule(rule, index) if rule is False or rule is True: # done or error return # check hierarchy if isinstance(rule, cssutils.css.CSSCharsetRule) or \ isinstance(rule, cssutils.css.CSSFontFaceRule) or \ isinstance(rule, cssutils.css.CSSImportRule) or \ isinstance(rule, cssutils.css.CSSNamespaceRule) or \ isinstance(rule, cssutils.css.CSSPageRule) or \ isinstance(rule, cssutils.css.MarginRule) or \ isinstance(rule, CSSMediaRule): self._log.error(u'%s: This type of rule is not allowed here: %s' % (self.__class__.__name__, rule.cssText), error=xml.dom.HierarchyRequestErr) return return self._finishInsertRule(rule, index) type = property(lambda self: self.MEDIA_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") wellformed = property(lambda self: self.media.wellformed) cssutils-1.0/src/cssutils/css/__init__.py0000666000175000017500000000355312126054542017265 0ustar hugohugo"""Implements Document Object Model Level 2 CSS http://www.w3.org/TR/2000/PR-DOM-Level-2-Style-20000927/css.html currently implemented - CSSStyleSheet - CSSRuleList - CSSRule - CSSComment (cssutils addon) - CSSCharsetRule - CSSFontFaceRule - CSSImportRule - CSSMediaRule - CSSNamespaceRule (WD) - CSSPageRule - CSSStyleRule - CSSUnkownRule - Selector and SelectorList - CSSStyleDeclaration - CSS2Properties - CSSValue - CSSPrimitiveValue - CSSValueList - CSSVariablesRule - CSSVariablesDeclaration todo - RGBColor, Rect, Counter """ __all__ = [ 'CSSStyleSheet', 'CSSRuleList', 'CSSRule', 'CSSComment', 'CSSCharsetRule', 'CSSFontFaceRule' 'CSSImportRule', 'CSSMediaRule', 'CSSNamespaceRule', 'CSSPageRule', 'MarginRule', 'CSSStyleRule', 'CSSUnknownRule', 'CSSVariablesRule', 'CSSVariablesDeclaration', 'Selector', 'SelectorList', 'CSSStyleDeclaration', 'Property', #'CSSValue', 'CSSPrimitiveValue', 'CSSValueList' 'PropertyValue', 'Value', 'ColorValue', 'DimensionValue', 'URIValue', 'CSSFunction', 'CSSVariable', 'MSValue' ] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssstylesheet import * from cssrulelist import * from cssrule import * from csscomment import * from csscharsetrule import * from cssfontfacerule import * from cssimportrule import * from cssmediarule import * from cssnamespacerule import * from csspagerule import * from marginrule import * from cssstylerule import * from cssvariablesrule import * from cssunknownrule import * from selector import * from selectorlist import * from cssstyledeclaration import * from cssvariablesdeclaration import * from property import * #from cssvalue import * from value import * cssutils-1.0/src/cssutils/css/csspagerule.py0000666000175000017500000003757412126054542020055 0ustar hugohugo"""CSSPageRule implements DOM Level 2 CSS CSSPageRule.""" __all__ = ['CSSPageRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from itertools import chain from cssstyledeclaration import CSSStyleDeclaration from marginrule import MarginRule import cssrule import cssutils import xml.dom class CSSPageRule(cssrule.CSSRuleRules): """ The CSSPageRule interface represents a @page rule within a CSS style sheet. The @page rule is used to specify the dimensions, orientation, margins, etc. of a page box for paged media. Format:: page : PAGE_SYM S* IDENT? pseudo_page? S* '{' S* [ declaration | margin ]? [ ';' S* [ declaration | margin ]? ]* '}' S* ; pseudo_page : ':' [ "left" | "right" | "first" ] ; margin : margin_sym S* '{' declaration [ ';' S* declaration? ]* '}' S* ; margin_sym : TOPLEFTCORNER_SYM | TOPLEFT_SYM | TOPCENTER_SYM | TOPRIGHT_SYM | TOPRIGHTCORNER_SYM | BOTTOMLEFTCORNER_SYM | BOTTOMLEFT_SYM | BOTTOMCENTER_SYM | BOTTOMRIGHT_SYM | BOTTOMRIGHTCORNER_SYM | LEFTTOP_SYM | LEFTMIDDLE_SYM | LEFTBOTTOM_SYM | RIGHTTOP_SYM | RIGHTMIDDLE_SYM | RIGHTBOTTOM_SYM ; `cssRules` contains a list of `MarginRule` objects. """ def __init__(self, selectorText=None, style=None, parentRule=None, parentStyleSheet=None, readonly=False): """ If readonly allows setting of properties in constructor only. :param selectorText: type string :param style: CSSStyleDeclaration for this CSSStyleRule """ super(CSSPageRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = u'@page' self._specificity = (0, 0, 0) tempseq = self._tempSeq() if selectorText: self.selectorText = selectorText tempseq.append(self.selectorText, 'selectorText') else: self._selectorText = self._tempSeq() if style: self.style = style else: self.style = CSSStyleDeclaration() tempseq.append(self.style, 'style') self._setSeq(tempseq) self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(selectorText=%r, style=%r)" % ( self.__class__.__name__, self.selectorText, self.style.cssText) def __str__(self): return (u"") % ( self.__class__.__name__, self.selectorText, self.specificity, self.style.cssText, len(self.cssRules), id(self)) def __contains__(self, margin): """Check if margin is set in the rule.""" return margin in self.keys() def keys(self): "Return list of all set margins (MarginRule)." return list(r.margin for r in self.cssRules) def __getitem__(self, margin): """Retrieve the style (of MarginRule) for `margin` (which must be normalized). """ for r in self.cssRules: if r.margin == margin: return r.style def __setitem__(self, margin, style): """Set the style (of MarginRule) for `margin` (which must be normalized). """ for i, r in enumerate(self.cssRules): if r.margin == margin: r.style = style return i else: return self.add(MarginRule(margin, style)) def __delitem__(self, margin): """Delete the style (the MarginRule) for `margin` (which must be normalized). """ for r in self.cssRules: if r.margin == margin: self.deleteRule(r) def __parseSelectorText(self, selectorText): """ Parse `selectorText` which may also be a list of tokens and returns (selectorText, seq). see _setSelectorText for details """ # for closures: must be a mutable new = {'wellformed': True, 'last-S': False, 'name': 0, 'first': 0, 'lr': 0} specificity = (0, 0, 0) def _char(expected, seq, token, tokenizer=None): # pseudo_page, :left, :right or :first val = self._tokenvalue(token) if not new['last-S'] and expected in ['page', ': or EOF']\ and u':' == val: try: identtoken = tokenizer.next() except StopIteration: self._log.error( u'CSSPageRule selectorText: No IDENT found.', token) else: ival, ityp = self._tokenvalue(identtoken),\ self._type(identtoken) if self._prods.IDENT != ityp: self._log.error(u'CSSPageRule selectorText: Expected ' u'IDENT but found: %r' % ival, token) else: if not ival in (u'first', u'left', u'right'): self._log.warn(u'CSSPageRule: Unknown @page ' u'selector: %r' % (u':'+ival,), neverraise=True) if ival == u'first': new['first'] = 1 else: new['lr'] = 1 seq.append(val + ival, 'pseudo') return 'EOF' return expected else: new['wellformed'] = False self._log.error(u'CSSPageRule selectorText: Unexpected CHAR: %r' % val, token) return expected def S(expected, seq, token, tokenizer=None): "Does not raise if EOF is found." if expected == ': or EOF': # pseudo must directly follow IDENT if given new['last-S'] = True return expected def IDENT(expected, seq, token, tokenizer=None): "" val = self._tokenvalue(token) if 'page' == expected: if self._normalize(val) == u'auto': self._log.error(u'CSSPageRule selectorText: Invalid pagename.', token) else: new['name'] = 1 seq.append(val, 'IDENT') return ': or EOF' else: new['wellformed'] = False self._log.error(u'CSSPageRule selectorText: Unexpected IDENT: ' u'%r' % val, token) return expected def COMMENT(expected, seq, token, tokenizer=None): "Does not raise if EOF is found." seq.append(cssutils.css.CSSComment([token]), 'COMMENT') return expected newseq = self._tempSeq() wellformed, expected = self._parse(expected='page', seq=newseq, tokenizer=self._tokenize2(selectorText), productions={'CHAR': _char, 'IDENT': IDENT, 'COMMENT': COMMENT, 'S': S}, new=new) wellformed = wellformed and new['wellformed'] # post conditions if expected == 'ident': self._log.error( u'CSSPageRule selectorText: No valid selector: %r' % self._valuestr(selectorText)) return wellformed, newseq, (new['name'], new['first'], new['lr']) def __parseMarginAndStyle(self, tokens): "tokens is a list, no generator (yet)" g = iter(tokens) styletokens = [] # new rules until parse done cssRules = [] for token in g: if token[0] == 'ATKEYWORD' and \ self._normalize(token[1]) in MarginRule.margins: # MarginRule m = MarginRule(parentRule=self, parentStyleSheet=self.parentStyleSheet) m.cssText = chain([token], g) # merge if margin set more than once for r in cssRules: if r.margin == m.margin: for p in m.style: r.style.setProperty(p, replace=False) break else: cssRules.append(m) continue # TODO: Properties? styletokens.append(token) return cssRules, styletokens def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSPageRule(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(CSSPageRule, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) if self._type(self._nexttoken(tokenizer)) != self._prods.PAGE_SYM: self._log.error(u'CSSPageRule: No CSSPageRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: newStyle = CSSStyleDeclaration(parentRule=self) ok = True selectortokens, startbrace = self._tokensupto2(tokenizer, blockstartonly=True, separateEnd=True) styletokens, braceorEOFtoken = self._tokensupto2(tokenizer, blockendonly=True, separateEnd=True) nonetoken = self._nexttoken(tokenizer) if self._tokenvalue(startbrace) != u'{': ok = False self._log.error(u'CSSPageRule: No start { of style declaration ' u'found: %r' % self._valuestr(cssText), startbrace) elif nonetoken: ok = False self._log.error(u'CSSPageRule: Trailing content found.', token=nonetoken) selok, newselseq, specificity = self.__parseSelectorText(selectortokens) ok = ok and selok val, type_ = self._tokenvalue(braceorEOFtoken),\ self._type(braceorEOFtoken) if val != u'}' and type_ != 'EOF': ok = False self._log.error( u'CSSPageRule: No "}" after style declaration found: %r' % self._valuestr(cssText)) else: if 'EOF' == type_: # add again as style needs it styletokens.append(braceorEOFtoken) # filter pagemargin rules out first cssRules, styletokens = self.__parseMarginAndStyle(styletokens) # SET, may raise: newStyle.cssText = styletokens if ok: self._selectorText = newselseq self._specificity = specificity self.style = newStyle self.cssRules = cssutils.css.CSSRuleList() for r in cssRules: self.cssRules.append(r) cssText = property(_getCssText, _setCssText, doc=u"(DOM) The parsable textual representation of this rule.") def _getSelectorText(self): """Wrapper for cssutils Selector object.""" return cssutils.ser.do_CSSPageRuleSelector(self._selectorText) def _setSelectorText(self, selectorText): """Wrapper for cssutils Selector object. :param selectorText: DOM String, in CSS 2.1 one of - :first - :left - :right - empty :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. """ self._checkReadonly() # may raise SYNTAX_ERR wellformed, newseq, specificity = self.__parseSelectorText(selectorText) if wellformed: self._selectorText = newseq self._specificity = specificity selectorText = property(_getSelectorText, _setSelectorText, doc=u"(DOM) The parsable textual representation of " u"the page selector for the rule.") def _setStyle(self, style): """ :param style: a CSSStyleDeclaration or string """ self._checkReadonly() if isinstance(style, basestring): self._style = CSSStyleDeclaration(cssText=style, parentRule=self) else: style._parentRule = self self._style = style style = property(lambda self: self._style, _setStyle, doc=u"(DOM) The declaration-block of this rule set, " u"a :class:`~cssutils.css.CSSStyleDeclaration`.") def insertRule(self, rule, index=None): """Implements base ``insertRule``.""" rule, index = self._prepareInsertRule(rule, index) if rule is False or rule is True: # done or error return # check hierarchy if isinstance(rule, cssutils.css.CSSCharsetRule) or \ isinstance(rule, cssutils.css.CSSFontFaceRule) or \ isinstance(rule, cssutils.css.CSSImportRule) or \ isinstance(rule, cssutils.css.CSSNamespaceRule) or \ isinstance(rule, CSSPageRule) or \ isinstance(rule, cssutils.css.CSSMediaRule): self._log.error(u'%s: This type of rule is not allowed here: %s' % (self.__class__.__name__, rule.cssText), error=xml.dom.HierarchyRequestErr) return return self._finishInsertRule(rule, index) specificity = property(lambda self: self._specificity, doc=u"""Specificity of this page rule (READONLY). Tuple of (f, g, h) where: - if the page selector has a named page, f=1; else f=0 - if the page selector has a ':first' pseudo-class, g=1; else g=0 - if the page selector has a ':left' or ':right' pseudo-class, h=1; else h=0 """) type = property(lambda self: self.PAGE_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") # constant but needed: wellformed = property(lambda self: True)cssutils-1.0/src/cssutils/css/marginrule.py0000666000175000017500000001707512126054542017677 0ustar hugohugo"""MarginRule implements DOM Level 2 CSS MarginRule.""" __all__ = ['MarginRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.prodparser import * from cssstyledeclaration import CSSStyleDeclaration import cssrule import cssutils import xml.dom class MarginRule(cssrule.CSSRule): """ A margin at-rule consists of an ATKEYWORD that identifies the margin box (e.g. '@top-left') and a block of declarations (said to be in the margin context). Format:: margin : margin_sym S* '{' declaration [ ';' S* declaration? ]* '}' S* ; margin_sym : TOPLEFTCORNER_SYM | TOPLEFT_SYM | TOPCENTER_SYM | TOPRIGHT_SYM | TOPRIGHTCORNER_SYM | BOTTOMLEFTCORNER_SYM | BOTTOMLEFT_SYM | BOTTOMCENTER_SYM | BOTTOMRIGHT_SYM | BOTTOMRIGHTCORNER_SYM | LEFTTOP_SYM | LEFTMIDDLE_SYM | LEFTBOTTOM_SYM | RIGHTTOP_SYM | RIGHTMIDDLE_SYM | RIGHTBOTTOM_SYM ; e.g.:: @top-left { content: "123"; } """ margins = ['@top-left-corner', '@top-left', '@top-center', '@top-right', '@top-right-corner', '@bottom-left-corner', '@bottom-left', '@bottom-center', '@bottom-right', '@bottom-right-corner', '@left-top', '@left-middle', '@left-bottom', '@right-top', '@right-middle', '@right-bottom' ] def __init__(self, margin=None, style=None, parentRule=None, parentStyleSheet=None, readonly=False): """ :param atkeyword: The margin area, e.g. '@top-left' for this rule :param style: CSSStyleDeclaration for this MarginRule """ super(MarginRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = self._keyword = None if margin: self.margin = margin if style: self.style = style else: self.style = CSSStyleDeclaration(parentRule=self) self._readonly = readonly def _setMargin(self, margin): """Check if new keyword fits the rule it is used for.""" n = self._normalize(margin) if n not in MarginRule.margins: self._log.error(u'Invalid margin @keyword for this %s rule: %r' % (self.margin, margin), error=xml.dom.InvalidModificationErr) else: self._atkeyword = n self._keyword = margin margin = property(lambda self: self._atkeyword, _setMargin, doc=u"Margin area of parent CSSPageRule. " u"`margin` and `atkeyword` are both normalized " u"@keyword of the @rule.") atkeyword = margin def __repr__(self): return u"cssutils.css.%s(margin=%r, style=%r)" % (self.__class__.__name__, self.margin, self.style.cssText) def __str__(self): return u"" % (self.__class__.__name__, self.margin, self.style.cssText, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_MarginRule(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(MarginRule, self)._setCssText(cssText) # TEMP: all style tokens are saved in store to fill styledeclaration # TODO: resolve when all generators styletokens = Prod(name='styletokens', match=lambda t, v: v != u'}', #toSeq=False, toStore='styletokens', storeToken=True ) prods = Sequence(Prod(name='@ margin', match=lambda t, v: t == 'ATKEYWORD' and self._normalize(v) in MarginRule.margins, toStore='margin' # TODO? #, exception=xml.dom.InvalidModificationErr ), PreDef.char('OPEN', u'{'), Sequence(Choice(PreDef.unknownrule(toStore='@'), styletokens), minmax=lambda: (0, None) ), PreDef.char('CLOSE', u'}', stopAndKeep=True) ) # parse ok, seq, store, unused = ProdParser().parse(cssText, u'MarginRule', prods) if ok: # TODO: use seq for serializing instead of fixed stuff? self._setSeq(seq) if 'margin' in store: # may raise: self.margin = store['margin'].value else: self._log.error(u'No margin @keyword for this %s rule' % self.margin, error=xml.dom.InvalidModificationErr) # new empty style self.style = CSSStyleDeclaration(parentRule=self) if 'styletokens' in store: # may raise: self.style.cssText = store['styletokens'] cssText = property(fget=_getCssText, fset=_setCssText, doc=u"(DOM) The parsable textual representation.") def _setStyle(self, style): """ :param style: A string or CSSStyleDeclaration which replaces the current style object. """ self._checkReadonly() if isinstance(style, basestring): self._style = CSSStyleDeclaration(cssText=style, parentRule=self) else: style._parentRule = self self._style = style style = property(lambda self: self._style, _setStyle, doc=u"(DOM) The declaration-block of this rule set.") type = property(lambda self: self.MARGIN_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") wellformed = property(lambda self: bool(self.atkeyword)) cssutils-1.0/src/cssutils/css/selector.py0000666000175000017500000010077412126054542017351 0ustar hugohugo"""Selector is a single Selector of a CSSStyleRule SelectorList. Partly implements http://www.w3.org/TR/css3-selectors/. TODO - .contains(selector) - .isSubselector(selector) """ __all__ = ['Selector'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.helper import Deprecated from cssutils.util import _SimpleNamespaces import cssutils import xml.dom class Selector(cssutils.util.Base2): """ (cssutils) a single selector in a :class:`~cssutils.css.SelectorList` of a :class:`~cssutils.css.CSSStyleRule`. Format:: # implemented in SelectorList selectors_group : selector [ COMMA S* selector ]* ; selector : simple_selector_sequence [ combinator simple_selector_sequence ]* ; combinator /* combinators can be surrounded by white space */ : PLUS S* | GREATER S* | TILDE S* | S+ ; simple_selector_sequence : [ type_selector | universal ] [ HASH | class | attrib | pseudo | negation ]* | [ HASH | class | attrib | pseudo | negation ]+ ; type_selector : [ namespace_prefix ]? element_name ; namespace_prefix : [ IDENT | '*' ]? '|' ; element_name : IDENT ; universal : [ namespace_prefix ]? '*' ; class : '.' IDENT ; attrib : '[' S* [ namespace_prefix ]? IDENT S* [ [ PREFIXMATCH | SUFFIXMATCH | SUBSTRINGMATCH | '=' | INCLUDES | DASHMATCH ] S* [ IDENT | STRING ] S* ]? ']' ; pseudo /* '::' starts a pseudo-element, ':' a pseudo-class */ /* Exceptions: :first-line, :first-letter, :before and :after. */ /* Note that pseudo-elements are restricted to one per selector and */ /* occur only in the last simple_selector_sequence. */ : ':' ':'? [ IDENT | functional_pseudo ] ; functional_pseudo : FUNCTION S* expression ')' ; expression /* In CSS3, the expressions are identifiers, strings, */ /* or of the form "an+b" */ : [ [ PLUS | '-' | DIMENSION | NUMBER | STRING | IDENT ] S* ]+ ; negation : NOT S* negation_arg S* ')' ; negation_arg : type_selector | universal | HASH | class | attrib | pseudo ; """ def __init__(self, selectorText=None, parent=None, readonly=False): """ :Parameters: selectorText initial value of this selector parent a SelectorList readonly default to False """ super(Selector, self).__init__() self.__namespaces = _SimpleNamespaces(log=self._log) self._element = None self._parent = parent self._specificity = (0, 0, 0, 0) if selectorText: self.selectorText = selectorText self._readonly = readonly def __repr__(self): if self.__getNamespaces(): st = (self.selectorText, self._getUsedNamespaces()) else: st = self.selectorText return u"cssutils.css.%s(selectorText=%r)" % (self.__class__.__name__, st) def __str__(self): return u"" % (self.__class__.__name__, self.selectorText, self.specificity, self._getUsedNamespaces(), id(self)) def _getUsedUris(self): "Return list of actually used URIs in this Selector." uris = set() for item in self.seq: type_, val = item.type, item.value if type_.endswith(u'-selector') or type_ == u'universal' and \ isinstance(val, tuple) and val[0] not in (None, u'*'): uris.add(val[0]) return uris def _getUsedNamespaces(self): "Return actually used namespaces only." useduris = self._getUsedUris() namespaces = _SimpleNamespaces(log=self._log) for p, uri in self._namespaces.items(): if uri in useduris: namespaces[p] = uri return namespaces def __getNamespaces(self): "Use own namespaces if not attached to a sheet, else the sheet's ones." try: return self._parent.parentRule.parentStyleSheet.namespaces except AttributeError: return self.__namespaces _namespaces = property(__getNamespaces, doc=u"If this Selector is attached to a " u"CSSStyleSheet the namespaces of that sheet " u"are mirrored here. While the Selector (or " u"parent SelectorList or parentRule(s) of that " u"are not attached a own dict of {prefix: " u"namespaceURI} is used.") element = property(lambda self: self._element, doc=u"Effective element target of this selector.") parent = property(lambda self: self._parent, doc=u"(DOM) The SelectorList that contains this Selector " u"or None if this Selector is not attached to a " u"SelectorList.") def _getSelectorText(self): """Return serialized format.""" return cssutils.ser.do_css_Selector(self) def _setSelectorText(self, selectorText): """ :param selectorText: parsable string or a tuple of (selectorText, dict-of-namespaces). Given namespaces are ignored if this object is attached to a CSSStyleSheet! :exceptions: - :exc:`~xml.dom.NamespaceErr`: Raised if the specified selector uses an unknown namespace prefix. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. """ self._checkReadonly() # might be (selectorText, namespaces) selectorText, namespaces = self._splitNamespacesOff(selectorText) try: # uses parent stylesheets namespaces if available, # otherwise given ones namespaces = self.parent.parentRule.parentStyleSheet.namespaces except AttributeError: pass tokenizer = self._tokenize2(selectorText) if not tokenizer: self._log.error(u'Selector: No selectorText given.') else: # prepare tokenlist: # "*" -> type "universal" # "*"|IDENT + "|" -> combined to "namespace_prefix" # "|" -> type "namespace_prefix" # "." + IDENT -> combined to "class" # ":" + IDENT, ":" + FUNCTION -> pseudo-class # FUNCTION "not(" -> negation # "::" + IDENT, "::" + FUNCTION -> pseudo-element tokens = [] for t in tokenizer: typ, val, lin, col = t if val == u':' and tokens and\ self._tokenvalue(tokens[-1]) == ':': # combine ":" and ":" tokens[-1] = (typ, u'::', lin, col) elif typ == 'IDENT' and tokens\ and self._tokenvalue(tokens[-1]) == u'.': # class: combine to .IDENT tokens[-1] = ('class', u'.'+val, lin, col) elif typ == 'IDENT' and tokens and \ self._tokenvalue(tokens[-1]).startswith(u':') and\ not self._tokenvalue(tokens[-1]).endswith(u'('): # pseudo-X: combine to :IDENT or ::IDENT but not ":a(" + "b" if self._tokenvalue(tokens[-1]).startswith(u'::'): t = 'pseudo-element' else: t = 'pseudo-class' tokens[-1] = (t, self._tokenvalue(tokens[-1])+val, lin, col) elif typ == 'FUNCTION' and val == u'not(' and tokens and \ u':' == self._tokenvalue(tokens[-1]): tokens[-1] = ('negation', u':' + val, lin, tokens[-1][3]) elif typ == 'FUNCTION' and tokens\ and self._tokenvalue(tokens[-1]).startswith(u':'): # pseudo-X: combine to :FUNCTION( or ::FUNCTION( if self._tokenvalue(tokens[-1]).startswith(u'::'): t = 'pseudo-element' else: t = 'pseudo-class' tokens[-1] = (t, self._tokenvalue(tokens[-1])+val, lin, col) elif val == u'*' and tokens and\ self._type(tokens[-1]) == 'namespace_prefix' and\ self._tokenvalue(tokens[-1]).endswith(u'|'): # combine prefix|* tokens[-1] = ('universal', self._tokenvalue(tokens[-1])+val, lin, col) elif val == u'*': # universal: "*" tokens.append(('universal', val, lin, col)) elif val == u'|' and tokens and\ self._type(tokens[-1]) in (self._prods.IDENT, 'universal')\ and self._tokenvalue(tokens[-1]).find(u'|') == -1: # namespace_prefix: "IDENT|" or "*|" tokens[-1] = ('namespace_prefix', self._tokenvalue(tokens[-1])+u'|', lin, col) elif val == u'|': # namespace_prefix: "|" tokens.append(('namespace_prefix', val, lin, col)) else: tokens.append(t) tokenizer = iter(tokens) # for closures: must be a mutable new = {'context': [''], # stack of: 'attrib', 'negation', 'pseudo' 'element': None, '_PREFIX': None, 'specificity': [0, 0, 0, 0], # mutable, finally a tuple! 'wellformed': True } # used for equality checks and setting of a space combinator S = u' ' def append(seq, val, typ=None, token=None): """ appends to seq namespace_prefix, IDENT will be combined to a tuple (prefix, name) where prefix might be None, the empty string or a prefix. Saved are also: - specificity definition: style, id, class/att, type - element: the element this Selector is for """ context = new['context'][-1] if token: line, col = token[2], token[3] else: line, col = None, None if typ == '_PREFIX': # SPECIAL TYPE: save prefix for combination with next new['_PREFIX'] = val[:-1] # handle next time return if new['_PREFIX'] is not None: # as saved from before and reset to None prefix, new['_PREFIX'] = new['_PREFIX'], None elif typ == 'universal' and '|' in val: # val == *|* or prefix|* prefix, val = val.split('|') else: prefix = None # namespace if (typ.endswith('-selector') or typ == 'universal') and not ( 'attribute-selector' == typ and not prefix): # att **IS NOT** in default ns if prefix == u'*': # *|name: in ANY_NS namespaceURI = cssutils._ANYNS elif prefix is None: # e or *: default namespace with prefix u'' # or local-name() namespaceURI = namespaces.get(u'', None) elif prefix == u'': # |name or |*: in no (or the empty) namespace namespaceURI = u'' else: # explicit namespace prefix # does not raise KeyError, see _SimpleNamespaces namespaceURI = namespaces[prefix] if namespaceURI is None: new['wellformed'] = False self._log.error(u'Selector: No namespaceURI found ' u'for prefix %r' % prefix, token=token, error=xml.dom.NamespaceErr) return # val is now (namespaceprefix, name) tuple val = (namespaceURI, val) # specificity if not context or context == 'negation': if 'id' == typ: new['specificity'][1] += 1 elif 'class' == typ or '[' == val: new['specificity'][2] += 1 elif typ in ('type-selector', 'negation-type-selector', 'pseudo-element'): new['specificity'][3] += 1 if not context and typ in ('type-selector', 'universal'): # define element new['element'] = val seq.append(val, typ, line=line, col=col) # expected constants simple_selector_sequence = 'type_selector universal HASH class ' \ 'attrib pseudo negation ' simple_selector_sequence2 = 'HASH class attrib pseudo negation ' element_name = 'element_name' negation_arg = 'type_selector universal HASH class attrib pseudo' negationend = ')' attname = 'prefix attribute' attname2 = 'attribute' attcombinator = 'combinator ]' # optional attvalue = 'value' # optional attend = ']' expressionstart = 'PLUS - DIMENSION NUMBER STRING IDENT' expression = expressionstart + ' )' combinator = ' combinator' def _COMMENT(expected, seq, token, tokenizer=None): "special implementation for comment token" append(seq, cssutils.css.CSSComment([token]), 'COMMENT', token=token) return expected def _S(expected, seq, token, tokenizer=None): # S context = new['context'][-1] if context.startswith('pseudo-'): if seq and seq[-1].value not in u'+-': # e.g. x:func(a + b) append(seq, S, 'S', token=token) return expected elif context != 'attrib' and 'combinator' in expected: append(seq, S, 'descendant', token=token) return simple_selector_sequence + combinator else: return expected def _universal(expected, seq, token, tokenizer=None): # *|* or prefix|* context = new['context'][-1] val = self._tokenvalue(token) if 'universal' in expected: append(seq, val, 'universal', token=token) if 'negation' == context: return negationend else: return simple_selector_sequence2 + combinator else: new['wellformed'] = False self._log.error( u'Selector: Unexpected universal.', token=token) return expected def _namespace_prefix(expected, seq, token, tokenizer=None): # prefix| => element_name # or prefix| => attribute_name if attrib context = new['context'][-1] val = self._tokenvalue(token) if 'attrib' == context and 'prefix' in expected: # [PREFIX|att] append(seq, val, '_PREFIX', token=token) return attname2 elif 'type_selector' in expected: # PREFIX|* append(seq, val, '_PREFIX', token=token) return element_name else: new['wellformed'] = False self._log.error( u'Selector: Unexpected namespace prefix.', token=token) return expected def _pseudo(expected, seq, token, tokenizer=None): # pseudo-class or pseudo-element :a ::a :a( ::a( """ /* '::' starts a pseudo-element, ':' a pseudo-class */ /* Exceptions: :first-line, :first-letter, :before and :after. */ /* Note that pseudo-elements are restricted to one per selector and */ /* occur only in the last simple_selector_sequence. */ """ context = new['context'][-1] val, typ = self._tokenvalue(token, normalize=True),\ self._type(token) if 'pseudo' in expected: if val in (':first-line', ':first-letter', ':before', ':after'): # always pseudo-element ??? typ = 'pseudo-element' append(seq, val, typ, token=token) if val.endswith(u'('): # function # "pseudo-" "class" or "element" new['context'].append(typ) return expressionstart elif 'negation' == context: return negationend elif 'pseudo-element' == typ: # only one per element, check at ) also! return combinator else: return simple_selector_sequence2 + combinator else: new['wellformed'] = False self._log.error( u'Selector: Unexpected start of pseudo.', token=token) return expected def _expression(expected, seq, token, tokenizer=None): # [ [ PLUS | '-' | DIMENSION | NUMBER | STRING | IDENT ] S* ]+ context = new['context'][-1] val, typ = self._tokenvalue(token), self._type(token) if context.startswith('pseudo-'): append(seq, val, typ, token=token) return expression else: new['wellformed'] = False self._log.error( u'Selector: Unexpected %s.' % typ, token=token) return expected def _attcombinator(expected, seq, token, tokenizer=None): # context: attrib # PREFIXMATCH | SUFFIXMATCH | SUBSTRINGMATCH | INCLUDES | # DASHMATCH context = new['context'][-1] val, typ = self._tokenvalue(token), self._type(token) if 'attrib' == context and 'combinator' in expected: # combinator in attrib append(seq, val, typ.lower(), token=token) return attvalue else: new['wellformed'] = False self._log.error( u'Selector: Unexpected %s.' % typ, token=token) return expected def _string(expected, seq, token, tokenizer=None): # identifier context = new['context'][-1] typ, val = self._type(token), self._stringtokenvalue(token) # context: attrib if 'attrib' == context and 'value' in expected: # attrib: [...=VALUE] append(seq, val, typ, token=token) return attend # context: pseudo elif context.startswith('pseudo-'): # :func(...) append(seq, val, typ, token=token) return expression else: new['wellformed'] = False self._log.error( u'Selector: Unexpected STRING.', token=token) return expected def _ident(expected, seq, token, tokenizer=None): # identifier context = new['context'][-1] val, typ = self._tokenvalue(token), self._type(token) # context: attrib if 'attrib' == context and 'attribute' in expected: # attrib: [...|ATT...] append(seq, val, 'attribute-selector', token=token) return attcombinator elif 'attrib' == context and 'value' in expected: # attrib: [...=VALUE] append(seq, val, 'attribute-value', token=token) return attend # context: negation elif 'negation' == context: # negation: (prefix|IDENT) append(seq, val, 'negation-type-selector', token=token) return negationend # context: pseudo elif context.startswith('pseudo-'): # :func(...) append(seq, val, typ, token=token) return expression elif 'type_selector' in expected or element_name == expected: # element name after ns or complete type_selector append(seq, val, 'type-selector', token=token) return simple_selector_sequence2 + combinator else: new['wellformed'] = False self._log.error(u'Selector: Unexpected IDENT.', token=token) return expected def _class(expected, seq, token, tokenizer=None): # .IDENT context = new['context'][-1] val = self._tokenvalue(token) if 'class' in expected: append(seq, val, 'class', token=token) if 'negation' == context: return negationend else: return simple_selector_sequence2 + combinator else: new['wellformed'] = False self._log.error(u'Selector: Unexpected class.', token=token) return expected def _hash(expected, seq, token, tokenizer=None): # #IDENT context = new['context'][-1] val = self._tokenvalue(token) if 'HASH' in expected: append(seq, val, 'id', token=token) if 'negation' == context: return negationend else: return simple_selector_sequence2 + combinator else: new['wellformed'] = False self._log.error(u'Selector: Unexpected HASH.', token=token) return expected def _char(expected, seq, token, tokenizer=None): # + > ~ ) [ ] + - context = new['context'][-1] val = self._tokenvalue(token) # context: attrib if u']' == val and 'attrib' == context and ']' in expected: # end of attrib append(seq, val, 'attribute-end', token=token) context = new['context'].pop() # attrib is done context = new['context'][-1] if 'negation' == context: return negationend else: return simple_selector_sequence2 + combinator elif u'=' == val and 'attrib' == context\ and 'combinator' in expected: # combinator in attrib append(seq, val, 'equals', token=token) return attvalue # context: negation elif u')' == val and 'negation' == context and u')' in expected: # not(negation_arg)" append(seq, val, 'negation-end', token=token) new['context'].pop() # negation is done context = new['context'][-1] return simple_selector_sequence + combinator # context: pseudo (at least one expression) elif val in u'+-' and context.startswith('pseudo-'): # :func(+ -)" _names = {'+': 'plus', '-': 'minus'} if val == u'+' and seq and seq[-1].value == S: seq.replace(-1, val, _names[val]) else: append(seq, val, _names[val], token=token) return expression elif u')' == val and context.startswith('pseudo-') and\ expression == expected: # :func(expression)" append(seq, val, 'function-end', token=token) new['context'].pop() # pseudo is done if 'pseudo-element' == context: return combinator else: return simple_selector_sequence + combinator # context: ROOT elif u'[' == val and 'attrib' in expected: # start of [attrib] append(seq, val, 'attribute-start', token=token) new['context'].append('attrib') return attname elif val in u'+>~' and 'combinator' in expected: # no other combinator except S may be following _names = { '>': 'child', '+': 'adjacent-sibling', '~': 'following-sibling'} if seq and seq[-1].value == S: seq.replace(-1, val, _names[val]) else: append(seq, val, _names[val], token=token) return simple_selector_sequence elif u',' == val: # not a selectorlist new['wellformed'] = False self._log.error( u'Selector: Single selector only.', error=xml.dom.InvalidModificationErr, token=token) return expected else: new['wellformed'] = False self._log.error( u'Selector: Unexpected CHAR.', token=token) return expected def _negation(expected, seq, token, tokenizer=None): # not( context = new['context'][-1] val = self._tokenvalue(token, normalize=True) if 'negation' in expected: new['context'].append('negation') append(seq, val, 'negation-start', token=token) return negation_arg else: new['wellformed'] = False self._log.error( u'Selector: Unexpected negation.', token=token) return expected def _atkeyword(expected, seq, token, tokenizer=None): "invalidates selector" new['wellformed'] = False self._log.error( u'Selector: Unexpected ATKEYWORD.', token=token) return expected # expected: only|not or mediatype, mediatype, feature, and newseq = self._tempSeq() wellformed, expected = self._parse( expected=simple_selector_sequence, seq=newseq, tokenizer=tokenizer, productions={'CHAR': _char, 'class': _class, 'HASH': _hash, 'STRING': _string, 'IDENT': _ident, 'namespace_prefix': _namespace_prefix, 'negation': _negation, 'pseudo-class': _pseudo, 'pseudo-element': _pseudo, 'universal': _universal, # pseudo 'NUMBER': _expression, 'DIMENSION': _expression, # attribute 'PREFIXMATCH': _attcombinator, 'SUFFIXMATCH': _attcombinator, 'SUBSTRINGMATCH': _attcombinator, 'DASHMATCH': _attcombinator, 'INCLUDES': _attcombinator, 'S': _S, 'COMMENT': _COMMENT, 'ATKEYWORD': _atkeyword}) wellformed = wellformed and new['wellformed'] # post condition if len(new['context']) > 1 or not newseq: wellformed = False self._log.error(u'Selector: Invalid or incomplete selector: %s' % self._valuestr(selectorText)) if expected == 'element_name': wellformed = False self._log.error(u'Selector: No element name found: %s' % self._valuestr(selectorText)) if expected == simple_selector_sequence and newseq: wellformed = False self._log.error(u'Selector: Cannot end with combinator: %s' % self._valuestr(selectorText)) if newseq and hasattr(newseq[-1].value, 'strip') \ and newseq[-1].value.strip() == u'': del newseq[-1] # set if wellformed: self.__namespaces = namespaces self._element = new['element'] self._specificity = tuple(new['specificity']) self._setSeq(newseq) # filter that only used ones are kept self.__namespaces = self._getUsedNamespaces() selectorText = property(_getSelectorText, _setSelectorText, doc=u"(DOM) The parsable textual representation of " u"the selector.") specificity = property(lambda self: self._specificity, doc="""Specificity of this selector (READONLY). Tuple of (a, b, c, d) where: a presence of style in document, always 0 if not used on a document b number of ID selectors c number of .class selectors d number of Element (type) selectors""") wellformed = property(lambda self: bool(len(self.seq))) @Deprecated('Use property parent instead') def _getParentList(self): return self.parent parentList = property(_getParentList, doc="DEPRECATED, see property parent instead") cssutils-1.0/src/cssutils/css/cssnamespacerule.py0000666000175000017500000002622012126054542021057 0ustar hugohugo"""CSSNamespaceRule currently implements http://dev.w3.org/csswg/css3-namespace/ """ __all__ = ['CSSNamespaceRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssrule import cssutils import xml.dom class CSSNamespaceRule(cssrule.CSSRule): """ Represents an @namespace rule within a CSS style sheet. The @namespace at-rule declares a namespace prefix and associates it with a given namespace (a string). This namespace prefix can then be used in namespace-qualified names such as those described in the Selectors Module [SELECT] or the Values and Units module [CSS3VAL]. Dealing with these rules directly is not needed anymore, easier is the use of :attr:`cssutils.css.CSSStyleSheet.namespaces`. Format:: namespace : NAMESPACE_SYM S* [namespace_prefix S*]? [STRING|URI] S* ';' S* ; namespace_prefix : IDENT ; """ def __init__(self, namespaceURI=None, prefix=None, cssText=None, parentRule=None, parentStyleSheet=None, readonly=False): """ :Parameters: namespaceURI The namespace URI (a simple string!) which is bound to the given prefix. If no prefix is set (``CSSNamespaceRule.prefix==''``) the namespace defined by namespaceURI is set as the default namespace prefix The prefix used in the stylesheet for the given ``CSSNamespaceRule.uri``. cssText if no namespaceURI is given cssText must be given to set a namespaceURI as this is readonly later on parentStyleSheet sheet where this rule belongs to Do not use as positional but as keyword parameters only! If readonly allows setting of properties in constructor only format namespace:: namespace : NAMESPACE_SYM S* [namespace_prefix S*]? [STRING|URI] S* ';' S* ; namespace_prefix : IDENT ; """ super(CSSNamespaceRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = u'@namespace' self._prefix = u'' self._namespaceURI = None if namespaceURI: self.namespaceURI = namespaceURI self.prefix = prefix tempseq = self._tempSeq() tempseq.append(self.prefix, 'prefix') tempseq.append(self.namespaceURI, 'namespaceURI') self._setSeq(tempseq) elif cssText is not None: self.cssText = cssText if parentStyleSheet: self._parentStyleSheet = parentStyleSheet self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(namespaceURI=%r, prefix=%r)" % ( self.__class__.__name__, self.namespaceURI, self.prefix) def __str__(self): return u"" % ( self.__class__.__name__, self.namespaceURI, self.prefix, id(self)) def _getCssText(self): """Return serialized property cssText""" return cssutils.ser.do_CSSNamespaceRule(self) def _setCssText(self, cssText): """ :param cssText: initial value for this rules cssText which is parsed :exceptions: - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. """ super(CSSNamespaceRule, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) attoken = self._nexttoken(tokenizer, None) if self._type(attoken) != self._prods.NAMESPACE_SYM: self._log.error(u'CSSNamespaceRule: No CSSNamespaceRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: # for closures: must be a mutable new = {'keyword': self._tokenvalue(attoken), 'prefix': u'', 'uri': None, 'wellformed': True } def _ident(expected, seq, token, tokenizer=None): # the namespace prefix, optional if 'prefix or uri' == expected: new['prefix'] = self._tokenvalue(token) seq.append(new['prefix'], 'prefix') return 'uri' else: new['wellformed'] = False self._log.error( u'CSSNamespaceRule: Unexpected ident.', token) return expected def _string(expected, seq, token, tokenizer=None): # the namespace URI as a STRING if expected.endswith('uri'): new['uri'] = self._stringtokenvalue(token) seq.append(new['uri'], 'namespaceURI') return ';' else: new['wellformed'] = False self._log.error( u'CSSNamespaceRule: Unexpected string.', token) return expected def _uri(expected, seq, token, tokenizer=None): # the namespace URI as URI which is DEPRECATED if expected.endswith('uri'): uri = self._uritokenvalue(token) new['uri'] = uri seq.append(new['uri'], 'namespaceURI') return ';' else: new['wellformed'] = False self._log.error( u'CSSNamespaceRule: Unexpected URI.', token) return expected def _char(expected, seq, token, tokenizer=None): # final ; val = self._tokenvalue(token) if ';' == expected and u';' == val: return 'EOF' else: new['wellformed'] = False self._log.error( u'CSSNamespaceRule: Unexpected char.', token) return expected # "NAMESPACE_SYM S* [namespace_prefix S*]? [STRING|URI] S* ';' S*" newseq = self._tempSeq() wellformed, expected = self._parse(expected='prefix or uri', seq=newseq, tokenizer=tokenizer, productions={'IDENT': _ident, 'STRING': _string, 'URI': _uri, 'CHAR': _char}, new=new) # wellformed set by parse wellformed = wellformed and new['wellformed'] # post conditions if new['uri'] is None: wellformed = False self._log.error(u'CSSNamespaceRule: No namespace URI found: %s' % self._valuestr(cssText)) if expected != 'EOF': wellformed = False self._log.error(u'CSSNamespaceRule: No ";" found: %s' % self._valuestr(cssText)) # set all if wellformed: self.atkeyword = new['keyword'] self._prefix = new['prefix'] self.namespaceURI = new['uri'] self._setSeq(newseq) cssText = property(fget=_getCssText, fset=_setCssText, doc=u"(DOM) The parsable textual representation of this " u"rule.") def _setNamespaceURI(self, namespaceURI): """ :param namespaceURI: the initial value for this rules namespaceURI :exceptions: - :exc:`~xml.dom.NoModificationAllowedErr`: (CSSRule) Raised if this rule is readonly or a namespaceURI is already set in this rule. """ self._checkReadonly() if not self._namespaceURI: # initial setting self._namespaceURI = namespaceURI tempseq = self._tempSeq() tempseq.append(namespaceURI, 'namespaceURI') self._setSeq(tempseq) # makes seq readonly! elif self._namespaceURI != namespaceURI: self._log.error(u'CSSNamespaceRule: namespaceURI is readonly.', error=xml.dom.NoModificationAllowedErr) namespaceURI = property(lambda self: self._namespaceURI, _setNamespaceURI, doc="URI (handled as simple string) of the defined namespace.") def _replaceNamespaceURI(self, namespaceURI): """Used during parse of new sheet only! :param namespaceURI: the new value for this rules namespaceURI """ self._namespaceURI = namespaceURI for i, x in enumerate(self._seq): if 'namespaceURI' == x.type: self._seq._readonly = False self._seq.replace(i, namespaceURI, 'namespaceURI') self._seq._readonly = True break def _setPrefix(self, prefix=None): """ :param prefix: the new prefix :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. """ self._checkReadonly() if not prefix: prefix = u'' else: tokenizer = self._tokenize2(prefix) prefixtoken = self._nexttoken(tokenizer, None) if not prefixtoken or self._type(prefixtoken) != self._prods.IDENT: self._log.error(u'CSSNamespaceRule: No valid prefix "%s".' % self._valuestr(prefix), error=xml.dom.SyntaxErr) return else: prefix = self._tokenvalue(prefixtoken) # update seq for i, x in enumerate(self._seq): if x == self._prefix: self._seq[i] = (prefix, 'prefix', None, None) break else: # put prefix at the beginning! self._seq[0] = (prefix, 'prefix', None, None) # set new prefix self._prefix = prefix prefix = property(lambda self: self._prefix, _setPrefix, doc=u"Prefix used for the defined namespace.") type = property(lambda self: self.NAMESPACE_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") wellformed = property(lambda self: self.namespaceURI is not None) cssutils-1.0/src/cssutils/css/cssrule.py0000666000175000017500000002773412126054542017215 0ustar hugohugo"""CSSRule implements DOM Level 2 CSS CSSRule.""" __all__ = ['CSSRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssutils import xml.dom class CSSRule(cssutils.util.Base2): """Abstract base interface for any type of CSS statement. This includes both rule sets and at-rules. An implementation is expected to preserve all rules specified in a CSS style sheet, even if the rule is not recognized by the parser. Unrecognized rules are represented using the :class:`CSSUnknownRule` interface. """ """ CSSRule type constants. An integer indicating which type of rule this is. """ UNKNOWN_RULE = 0 ":class:`cssutils.css.CSSUnknownRule` (not used in CSSOM anymore)" STYLE_RULE = 1 ":class:`cssutils.css.CSSStyleRule`" CHARSET_RULE = 2 ":class:`cssutils.css.CSSCharsetRule` (not used in CSSOM anymore)" IMPORT_RULE = 3 ":class:`cssutils.css.CSSImportRule`" MEDIA_RULE = 4 ":class:`cssutils.css.CSSMediaRule`" FONT_FACE_RULE = 5 ":class:`cssutils.css.CSSFontFaceRule`" PAGE_RULE = 6 ":class:`cssutils.css.CSSPageRule`" NAMESPACE_RULE = 10 """:class:`cssutils.css.CSSNamespaceRule`, Value has changed in 0.9.7a3 due to a change in the CSSOM spec.""" COMMENT = 1001 # was -1, cssutils only """:class:`cssutils.css.CSSComment` - not in the offical spec, Value has changed in 0.9.7a3""" VARIABLES_RULE = 1008 """:class:`cssutils.css.CSSVariablesRule` - experimental rule not in the offical spec""" MARGIN_RULE = 1006 """:class:`cssutils.css.MarginRule` - experimental rule not in the offical spec""" _typestrings = {UNKNOWN_RULE: u'UNKNOWN_RULE', STYLE_RULE: u'STYLE_RULE', CHARSET_RULE: u'CHARSET_RULE', IMPORT_RULE: u'IMPORT_RULE', MEDIA_RULE: u'MEDIA_RULE', FONT_FACE_RULE: u'FONT_FACE_RULE', PAGE_RULE: u'PAGE_RULE', NAMESPACE_RULE: u'NAMESPACE_RULE', COMMENT: u'COMMENT', VARIABLES_RULE: u'VARIABLES_RULE', MARGIN_RULE: u'MARGIN_RULE' } def __init__(self, parentRule=None, parentStyleSheet=None, readonly=False): """Set common attributes for all rules.""" super(CSSRule, self).__init__() self._parent = parentRule self._parentRule = parentRule self._parentStyleSheet = parentStyleSheet self._setSeq(self._tempSeq()) #self._atkeyword = None # must be set after initialization of #inheriting rule is done self._readonly = False def _setAtkeyword(self, keyword): """Check if new keyword fits the rule it is used for.""" atkeyword = self._normalize(keyword) if not self.atkeyword or (self.atkeyword == atkeyword): self._atkeyword = atkeyword self._keyword = keyword else: self._log.error(u'%s: Invalid atkeyword for this rule: %r' % (self.atkeyword, keyword), error=xml.dom.InvalidModificationErr) atkeyword = property(lambda self: self._atkeyword, _setAtkeyword, doc=u"Normalized keyword of an @rule (e.g. ``@import``).") def _setCssText(self, cssText): """ :param cssText: A parsable DOMString. :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ self._checkReadonly() cssText = property(lambda self: u'', _setCssText, doc=u"(DOM) The parsable textual representation of the " u"rule. This reflects the current state of the rule " u"and not its initial value.") parent = property(lambda self: self._parent, doc=u"The Parent Node of this CSSRule or None.") parentRule = property(lambda self: self._parentRule, doc=u"If this rule is contained inside another rule " u"(e.g. a style rule inside an @media block), this " u"is the containing rule. If this rule is not nested " u"inside any other rules, this returns None.") def _getParentStyleSheet(self): # rules contained in other rules (@media) use that rules parent if (self.parentRule): return self.parentRule._parentStyleSheet else: return self._parentStyleSheet parentStyleSheet = property(_getParentStyleSheet, doc=u"The style sheet that contains this rule.") type = property(lambda self: self.UNKNOWN_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") typeString = property(lambda self: CSSRule._typestrings[self.type], doc=u"Descriptive name of this rule's type.") wellformed = property(lambda self: False, doc=u"If the rule is wellformed.") class CSSRuleRules(CSSRule): """Abstract base interface for rules that contain other rules like @media or @page. Methods may be overwritten if a rule has specific stuff to do like checking the order of insertion like @media does. """ def __init__(self, parentRule=None, parentStyleSheet=None): super(CSSRuleRules, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self.cssRules = cssutils.css.CSSRuleList() def __iter__(self): """Generator iterating over these rule's cssRules.""" for rule in self._cssRules: yield rule def _setCssRules(self, cssRules): "Set new cssRules and update contained rules refs." cssRules.append = self.insertRule cssRules.extend = self.insertRule cssRules.__delitem__ == self.deleteRule for rule in cssRules: rule._parentRule = self rule._parentStyleSheet = None self._cssRules = cssRules cssRules = property(lambda self: self._cssRules, _setCssRules, "All Rules in this style sheet, a " ":class:`~cssutils.css.CSSRuleList`.") def deleteRule(self, index): """ Delete the rule at `index` from rules ``cssRules``. :param index: The `index` of the rule to be removed from the rules cssRules list. For an `index` < 0 **no** :exc:`~xml.dom.IndexSizeErr` is raised but rules for normal Python lists are used. E.g. ``deleteRule(-1)`` removes the last rule in cssRules. `index` may also be a CSSRule object which will then be removed. :Exceptions: - :exc:`~xml.dom.IndexSizeErr`: Raised if the specified index does not correspond to a rule in the media rule list. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this media rule is readonly. """ self._checkReadonly() if isinstance(index, CSSRule): for i, r in enumerate(self.cssRules): if index == r: index = i break else: raise xml.dom.IndexSizeErr(u"%s: Not a rule in " u"this rule'a cssRules list: %s" % (self.__class__.__name__, index)) try: # detach self._cssRules[index]._parentRule = None del self._cssRules[index] except IndexError: raise xml.dom.IndexSizeErr(u'%s: %s is not a valid index ' u'in the rulelist of length %i' % (self.__class__.__name__, index, self._cssRules.length)) def _prepareInsertRule(self, rule, index=None): "return checked `index` and optional parsed `rule`" self._checkReadonly() # check index if index is None: index = len(self._cssRules) elif index < 0 or index > self._cssRules.length: raise xml.dom.IndexSizeErr(u'%s: Invalid index %s for ' u'CSSRuleList with a length of %s.' % (self.__class__.__name__, index, self._cssRules.length)) # check and optionally parse rule if isinstance(rule, basestring): tempsheet = cssutils.css.CSSStyleSheet() tempsheet.cssText = rule if len(tempsheet.cssRules) != 1 or (tempsheet.cssRules and not isinstance(tempsheet.cssRules[0], cssutils.css.CSSRule)): self._log.error(u'%s: Invalid Rule: %s' % (self.__class__.__name__, rule)) return False, False rule = tempsheet.cssRules[0] elif isinstance(rule, cssutils.css.CSSRuleList): # insert all rules for i, r in enumerate(rule): self.insertRule(r, index + i) return True, True elif not isinstance(rule, cssutils.css.CSSRule): self._log.error(u'%s: Not a CSSRule: %s' % (rule, self.__class__.__name__)) return False, False return rule, index def _finishInsertRule(self, rule, index): "add `rule` at `index`" rule._parentRule = self rule._parentStyleSheet = None self._cssRules.insert(index, rule) return index def add(self, rule): """Add `rule` to page rule. Same as ``insertRule(rule)``.""" return self.insertRule(rule) def insertRule(self, rule, index=None): """ Insert `rule` into the rules ``cssRules``. :param rule: the parsable text representing the `rule` to be inserted. For rule sets this contains both the selector and the style declaration. For at-rules, this specifies both the at-identifier and the rule content. cssutils also allows rule to be a valid :class:`~cssutils.css.CSSRule` object. :param index: before the `index` the specified `rule` will be inserted. If the specified `index` is equal to the length of the rules rule collection, the rule will be added to the end of the rule. If index is not given or None rule will be appended to rule list. :returns: the index of the newly inserted rule. :exceptions: - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the `rule` cannot be inserted at the specified `index`, e.g., if an @import rule is inserted after a standard rule set or other at-rule. - :exc:`~xml.dom.IndexSizeErr`: Raised if the specified `index` is not a valid insertion point. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified `rule` has a syntax error and is unparsable. """ return self._prepareInsertRule(rule, index) cssutils-1.0/src/cssutils/css/property.py0000666000175000017500000004452212126111116017402 0ustar hugohugo"""Property is a single CSS property in a CSSStyleDeclaration.""" __all__ = ['Property'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.helper import Deprecated from value import PropertyValue import cssutils import xml.dom class Property(cssutils.util.Base): """A CSS property in a StyleDeclaration of a CSSStyleRule (cssutils). Format:: property = name : IDENT S* ; expr = value : term [ operator term ]* ; term : unary_operator? [ NUMBER S* | PERCENTAGE S* | LENGTH S* | EMS S* | EXS S* | ANGLE S* | TIME S* | FREQ S* | function ] | STRING S* | IDENT S* | URI S* | hexcolor ; function : FUNCTION S* expr ')' S* ; /* * There is a constraint on the color that it must * have either 3 or 6 hex-digits (i.e., [0-9a-fA-F]) * after the "#"; e.g., "#000" is OK, but "#abcd" is not. */ hexcolor : HASH S* ; prio : IMPORTANT_SYM S* ; """ def __init__(self, name=None, value=None, priority=u'', _mediaQuery=False, parent=None): """ :param name: a property name string (will be normalized) :param value: a property value string :param priority: an optional priority string which currently must be u'', u'!important' or u'important' :param _mediaQuery: if ``True`` value is optional (used by MediaQuery) :param parent: the parent object, normally a :class:`cssutils.css.CSSStyleDeclaration` """ super(Property, self).__init__() self.seqs = [[], None, []] self.wellformed = False self._mediaQuery = _mediaQuery self.parent = parent self.__nametoken = None self._name = u'' self._literalname = u'' self.seqs[1] = PropertyValue(parent=self) if name: self.name = name self.propertyValue = value self._priority = u'' self._literalpriority = u'' if priority: self.priority = priority def __repr__(self): return u"cssutils.css.%s(name=%r, value=%r, priority=%r)" % ( self.__class__.__name__, self.literalname, self.propertyValue.cssText, self.priority) def __str__(self): return u"<%s.%s object name=%r value=%r priority=%r valid=%r at 0x%x>" \ % (self.__class__.__module__, self.__class__.__name__, self.name, self.propertyValue.cssText, self.priority, self.valid, id(self)) def _isValidating(self): """Return True if validation is enabled.""" try: return self.parent.validating except AttributeError: # default (no parent) return True def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_Property(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ # check and prepare tokenlists for setting tokenizer = self._tokenize2(cssText) nametokens = self._tokensupto2(tokenizer, propertynameendonly=True) if nametokens: wellformed = True valuetokens = self._tokensupto2(tokenizer, propertyvalueendonly=True) prioritytokens = self._tokensupto2(tokenizer, propertypriorityendonly=True) if self._mediaQuery and not valuetokens: # MediaQuery may consist of name only self.name = nametokens self.propertyValue = None self.priority = None return # remove colon from nametokens colontoken = nametokens.pop() if self._tokenvalue(colontoken) != u':': wellformed = False self._log.error(u'Property: No ":" after name found: %s' % self._valuestr(cssText), colontoken) elif not nametokens: wellformed = False self._log.error(u'Property: No property name found: %s' % self._valuestr(cssText), colontoken) if valuetokens: if self._tokenvalue(valuetokens[-1]) == u'!': # priority given, move "!" to prioritytokens prioritytokens.insert(0, valuetokens.pop(-1)) else: wellformed = False self._log.error(u'Property: No property value found: %s' % self._valuestr(cssText), colontoken) if wellformed: self.wellformed = True self.name = nametokens self.propertyValue = valuetokens self.priority = prioritytokens # also invalid values are set! if self._isValidating(): self.validate() else: self._log.error(u'Property: No property name found: %s' % self._valuestr(cssText)) cssText = property(fget=_getCssText, fset=_setCssText, doc="A parsable textual representation.") def _setName(self, name): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified name has a syntax error and is unparsable. """ # for closures: must be a mutable new = {'literalname': None, 'wellformed': True} def _ident(expected, seq, token, tokenizer=None): # name if 'name' == expected: new['literalname'] = self._tokenvalue(token).lower() seq.append(new['literalname']) return 'EOF' else: new['wellformed'] = False self._log.error(u'Property: Unexpected ident.', token) return expected newseq = [] wellformed, expected = self._parse(expected='name', seq=newseq, tokenizer=self._tokenize2(name), productions={'IDENT': _ident}) wellformed = wellformed and new['wellformed'] # post conditions # define a token for error logging if isinstance(name, list): token = name[0] self.__nametoken = token else: token = None if not new['literalname']: wellformed = False self._log.error(u'Property: No name found: %s' % self._valuestr(name), token=token) if wellformed: self.wellformed = True self._literalname = new['literalname'] self._name = self._normalize(self._literalname) self.seqs[0] = newseq # validate if self._isValidating() and self._name not in cssutils.profile.knownNames: # self.valid = False self._log.warn(u'Property: Unknown Property name.', token=token, neverraise=True) else: pass # self.valid = True # if self.propertyValue: # self.propertyValue._propertyName = self._name # #self.valid = self.propertyValue.valid else: self.wellformed = False name = property(lambda self: self._name, _setName, doc="Name of this property.") literalname = property(lambda self: self._literalname, doc="Readonly literal (not normalized) name " "of this property") def _setPropertyValue(self, cssText): """ See css.PropertyValue :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error (according to the attached property) or is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: TODO: Raised if the specified CSS string value represents a different type of values than the values allowed by the CSS property. """ if self._mediaQuery and not cssText: self.seqs[1] = PropertyValue(parent=self) else: self.seqs[1].cssText = cssText self.wellformed = self.wellformed and self.seqs[1].wellformed propertyValue = property(lambda self: self.seqs[1], _setPropertyValue, doc=u"(cssutils) PropertyValue object of property") def _getValue(self): if self.propertyValue: # value without comments return self.propertyValue.value else: return u'' def _setValue(self, value): self._setPropertyValue(value) value = property(_getValue, _setValue, doc="The textual value of this Properties propertyValue.") def _setPriority(self, priority): """ priority a string, currently either u'', u'!important' or u'important' Format:: prio : IMPORTANT_SYM S* ; "!"{w}"important" {return IMPORTANT_SYM;} :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified priority has a syntax error and is unparsable. In this case a priority not equal to None, "" or "!{w}important". As CSSOM defines CSSStyleDeclaration.getPropertyPriority resulting in u'important' this value is also allowed to set a Properties priority """ if self._mediaQuery: self._priority = u'' self._literalpriority = u'' if priority: self._log.error(u'Property: No priority in a MediaQuery - ' u'ignored.') return if isinstance(priority, basestring) and\ u'important' == self._normalize(priority): priority = u'!%s' % priority # for closures: must be a mutable new = {'literalpriority': u'', 'wellformed': True} def _char(expected, seq, token, tokenizer=None): # "!" val = self._tokenvalue(token) if u'!' == expected == val: seq.append(val) return 'important' else: new['wellformed'] = False self._log.error(u'Property: Unexpected char.', token) return expected def _ident(expected, seq, token, tokenizer=None): # "important" val = self._tokenvalue(token) if 'important' == expected: new['literalpriority'] = val seq.append(val) return 'EOF' else: new['wellformed'] = False self._log.error(u'Property: Unexpected ident.', token) return expected newseq = [] wellformed, expected = self._parse(expected='!', seq=newseq, tokenizer=self._tokenize2(priority), productions={'CHAR': _char, 'IDENT': _ident}) wellformed = wellformed and new['wellformed'] # post conditions if priority and not new['literalpriority']: wellformed = False self._log.info(u'Property: Invalid priority: %s' % self._valuestr(priority)) if wellformed: self.wellformed = self.wellformed and wellformed self._literalpriority = new['literalpriority'] self._priority = self._normalize(self.literalpriority) self.seqs[2] = newseq # validate priority if self._priority not in (u'', u'important'): self._log.error(u'Property: No CSS priority value: %s' % self._priority) priority = property(lambda self: self._priority, _setPriority, doc="Priority of this property.") literalpriority = property(lambda self: self._literalpriority, doc="Readonly literal (not normalized) priority of this property") def _setParent(self, parent): self._parent = parent parent = property(lambda self: self._parent, _setParent, doc="The Parent Node (normally a CSSStyledeclaration) of this " "Property") def validate(self): """Validate value against `profiles` which are checked dynamically. properties in e.g. @font-face rules are checked against ``cssutils.profile.CSS3_FONT_FACE`` only. For each of the following cases a message is reported: - INVALID (so the property is known but not valid) ``ERROR Property: Invalid value for "{PROFILE-1[/PROFILE-2...]" property: ...`` - VALID but not in given profiles or defaultProfiles ``WARNING Property: Not valid for profile "{PROFILE-X}" but valid "{PROFILE-Y}" property: ...`` - VALID in current profile ``DEBUG Found valid "{PROFILE-1[/PROFILE-2...]" property...`` - UNKNOWN property ``WARNING Unknown Property name...`` is issued so for example:: cssutils.log.setLevel(logging.DEBUG) parser = cssutils.CSSParser() s = parser.parseString('''body { unknown-property: x; color: 4; color: rgba(1,2,3,4); color: red }''') # Log output: WARNING Property: Unknown Property name. [2:9: unknown-property] ERROR Property: Invalid value for "CSS Color Module Level 3/CSS Level 2.1" property: 4 [3:9: color] DEBUG Property: Found valid "CSS Color Module Level 3" value: rgba(1, 2, 3, 4) [4:9: color] DEBUG Property: Found valid "CSS Level 2.1" value: red [5:9: color] and when setting an explicit default profile:: cssutils.profile.defaultProfiles = cssutils.profile.CSS_LEVEL_2 s = parser.parseString('''body { unknown-property: x; color: 4; color: rgba(1,2,3,4); color: red }''') # Log output: WARNING Property: Unknown Property name. [2:9: unknown-property] ERROR Property: Invalid value for "CSS Color Module Level 3/CSS Level 2.1" property: 4 [3:9: color] WARNING Property: Not valid for profile "CSS Level 2.1" but valid "CSS Color Module Level 3" value: rgba(1, 2, 3, 4) [4:9: color] DEBUG Property: Found valid "CSS Level 2.1" value: red [5:9: color] """ valid = False profiles = None try: # if @font-face use that profile rule = self.parent.parentRule except AttributeError: pass else: if rule is not None: if rule.type == rule.FONT_FACE_RULE: profiles = [cssutils.profile.CSS3_FONT_FACE] #TODO: same for @page if self.name and self.value: cv = self.propertyValue # TODO # if cv.cssValueType == cv.CSS_VARIABLE and not cv.value: # # TODO: false alarms too! # cssutils.log.warn(u'No value for variable "%s" found, keeping ' # u'variable.' % cv.name, neverraise=True) if self.name in cssutils.profile.knownNames: # add valid, matching, validprofiles... valid, matching, validprofiles = \ cssutils.profile.validateWithProfile(self.name, self.value, profiles) if not valid: self._log.error(u'Property: Invalid value for ' u'"%s" property: %s' % (u'/'.join(validprofiles), self.value), token=self.__nametoken, neverraise=True) # TODO: remove logic to profiles! elif valid and not matching:#(profiles and profiles not in validprofiles): if not profiles: notvalidprofiles = u'/'.join(cssutils.profile.defaultProfiles) else: notvalidprofiles = profiles self._log.warn(u'Property: Not valid for profile "%s" ' u'but valid "%s" value: %s ' % (notvalidprofiles, u'/'.join(validprofiles), self.value), token = self.__nametoken, neverraise=True) valid = False elif valid: self._log.debug(u'Property: Found valid "%s" value: %s' % (u'/'.join(validprofiles), self.value), token = self.__nametoken, neverraise=True) if self._priority not in (u'', u'important'): valid = False return valid valid = property(validate, doc=u"Check if value of this property is valid " u"in the properties context.") @Deprecated(u'Use ``property.propertyValue`` instead.') def _getCSSValue(self): return self.propertyValue @Deprecated(u'Use ``property.propertyValue`` instead.') def _setCSSValue(self, cssText): self._setPropertyValue(cssText) cssValue = property(_getCSSValue, _setCSSValue, doc="(DEPRECATED) Use ``property.propertyValue`` instead.") cssutils-1.0/src/cssutils/css/cssunknownrule.py0000666000175000017500000002036312126054542020624 0ustar hugohugo"""CSSUnknownRule implements DOM Level 2 CSS CSSUnknownRule.""" __all__ = ['CSSUnknownRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssrule import cssutils import xml.dom class CSSUnknownRule(cssrule.CSSRule): """ Represents an at-rule not supported by this user agent, so in effect all other at-rules not defined in cssutils. Format:: @xxx until ';' or block {...} """ def __init__(self, cssText=u'', parentRule=None, parentStyleSheet=None, readonly=False): """ :param cssText: of type string """ super(CSSUnknownRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = None if cssText: self.cssText = cssText self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(cssText=%r)" % ( self.__class__.__name__, self.cssText) def __str__(self): return u"" % ( self.__class__.__name__, self.cssText, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSUnknownRule(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(CSSUnknownRule, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) attoken = self._nexttoken(tokenizer, None) if not attoken or self._type(attoken) != self._prods.ATKEYWORD: self._log.error(u'CSSUnknownRule: No CSSUnknownRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: # for closures: must be a mutable new = {'nesting': [], # {} [] or () 'wellformed': True } def CHAR(expected, seq, token, tokenizer=None): type_, val, line, col = token if expected != 'EOF': if val in u'{[(': new['nesting'].append(val) elif val in u'}])': opening = {u'}': u'{', u']': u'[', u')': u'('}[val] try: if new['nesting'][-1] == opening: new['nesting'].pop() else: raise IndexError() except IndexError: new['wellformed'] = False self._log.error(u'CSSUnknownRule: Wrong nesting of ' u'{, [ or (.', token=token) if val in u'};' and not new['nesting']: expected = 'EOF' seq.append(val, type_, line=line, col=col) return expected else: new['wellformed'] = False self._log.error(u'CSSUnknownRule: Expected end of rule.', token=token) return expected def FUNCTION(expected, seq, token, tokenizer=None): # handled as opening ( type_, val, line, col = token val = self._tokenvalue(token) if expected != 'EOF': new['nesting'].append(u'(') seq.append(val, type_, line=line, col=col) return expected else: new['wellformed'] = False self._log.error(u'CSSUnknownRule: Expected end of rule.', token=token) return expected def EOF(expected, seq, token, tokenizer=None): "close all blocks and return 'EOF'" for x in reversed(new['nesting']): closing = {u'{': u'}', u'[': u']', u'(': u')'}[x] seq.append(closing, closing) new['nesting'] = [] return 'EOF' def INVALID(expected, seq, token, tokenizer=None): # makes rule invalid self._log.error(u'CSSUnknownRule: Bad syntax.', token=token, error=xml.dom.SyntaxErr) new['wellformed'] = False return expected def STRING(expected, seq, token, tokenizer=None): type_, val, line, col = token val = self._stringtokenvalue(token) if expected != 'EOF': seq.append(val, type_, line=line, col=col) return expected else: new['wellformed'] = False self._log.error(u'CSSUnknownRule: Expected end of rule.', token=token) return expected def URI(expected, seq, token, tokenizer=None): type_, val, line, col = token val = self._uritokenvalue(token) if expected != 'EOF': seq.append(val, type_, line=line, col=col) return expected else: new['wellformed'] = False self._log.error(u'CSSUnknownRule: Expected end of rule.', token=token) return expected def default(expected, seq, token, tokenizer=None): type_, val, line, col = token if expected != 'EOF': seq.append(val, type_, line=line, col=col) return expected else: new['wellformed'] = False self._log.error(u'CSSUnknownRule: Expected end of rule.', token=token) return expected # unknown : ATKEYWORD S* ... ; | } newseq = self._tempSeq() wellformed, expected = self._parse(expected=None, seq=newseq, tokenizer=tokenizer, productions={'CHAR': CHAR, 'EOF': EOF, 'FUNCTION': FUNCTION, 'INVALID': INVALID, 'STRING': STRING, 'URI': URI, 'S': default # overwrite default default! }, default=default, new=new) # wellformed set by parse wellformed = wellformed and new['wellformed'] # post conditions if expected != 'EOF': wellformed = False self._log.error(u'CSSUnknownRule: No ending ";" or "}" found: ' u'%r' % self._valuestr(cssText)) elif new['nesting']: wellformed = False self._log.error(u'CSSUnknownRule: Unclosed "{", "[" or "(": %r' % self._valuestr(cssText)) # set all if wellformed: self.atkeyword = self._tokenvalue(attoken) self._setSeq(newseq) cssText = property(fget=_getCssText, fset=_setCssText, doc=u"(DOM) The parsable textual representation.") type = property(lambda self: self.UNKNOWN_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") wellformed = property(lambda self: bool(self.atkeyword)) cssutils-1.0/src/cssutils/css/cssproperties.py0000666000175000017500000001207712126054542020434 0ustar hugohugo"""CSS2Properties (partly!) implements DOM Level 2 CSS CSS2Properties used by CSSStyleDeclaration TODO: CSS2Properties If an implementation does implement this interface, it is expected to understand the specific syntax of the shorthand properties, and apply their semantics; when the margin property is set, for example, the marginTop, marginRight, marginBottom and marginLeft properties are actually being set by the underlying implementation. When dealing with CSS "shorthand" properties, the shorthand properties should be decomposed into their component longhand properties as appropriate, and when querying for their value, the form returned should be the shortest form exactly equivalent to the declarations made in the ruleset. However, if there is no shorthand declaration that could be added to the ruleset without changing in any way the rules already declared in the ruleset (i.e., by adding longhand rules that were previously not declared in the ruleset), then the empty string should be returned for the shorthand property. For example, querying for the font property should not return "normal normal normal 14pt/normal Arial, sans-serif", when "14pt Arial, sans-serif" suffices. (The normals are initial values, and are implied by use of the longhand property.) If the values for all the longhand properties that compose a particular string are the initial values, then a string consisting of all the initial values should be returned (e.g. a border-width value of "medium" should be returned as such, not as ""). For some shorthand properties that take missing values from other sides, such as the margin, padding, and border-[width|style|color] properties, the minimum number of sides possible should be used; i.e., "0px 10px" will be returned instead of "0px 10px 0px 10px". If the value of a shorthand property can not be decomposed into its component longhand properties, as is the case for the font property with a value of "menu", querying for the values of the component longhand properties should return the empty string. TODO: CSS2Properties DOMImplementation The interface found within this section are not mandatory. A DOM application can use the hasFeature method of the DOMImplementation interface to determine whether it is supported or not. The feature string for this extended interface listed in this section is "CSS2" and the version is "2.0". """ __all__ = ['CSS2Properties'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssutils.profiles import re class CSS2Properties(object): """The CSS2Properties interface represents a convenience mechanism for retrieving and setting properties within a CSSStyleDeclaration. The attributes of this interface correspond to all the properties specified in CSS2. Getting an attribute of this interface is equivalent to calling the getPropertyValue method of the CSSStyleDeclaration interface. Setting an attribute of this interface is equivalent to calling the setProperty method of the CSSStyleDeclaration interface. cssutils actually also allows usage of ``del`` to remove a CSS property from a CSSStyleDeclaration. This is an abstract class, the following functions need to be present in inheriting class: - ``_getP`` - ``_setP`` - ``_delP`` """ # actual properties are set after the class definition! def _getP(self, CSSname): pass def _setP(self, CSSname, value): pass def _delP(self, CSSname): pass _reCSStoDOMname = re.compile('-[a-z]', re.I) def _toDOMname(CSSname): """Returns DOMname for given CSSname e.g. for CSSname 'font-style' returns 'fontStyle'. """ def _doCSStoDOMname2(m): return m.group(0)[1].capitalize() return _reCSStoDOMname.sub(_doCSStoDOMname2, CSSname) _reDOMtoCSSname = re.compile('([A-Z])[a-z]+') def _toCSSname(DOMname): """Return CSSname for given DOMname e.g. for DOMname 'fontStyle' returns 'font-style'. """ def _doDOMtoCSSname2(m): return '-' + m.group(0).lower() return _reDOMtoCSSname.sub(_doDOMtoCSSname2, DOMname) # add list of DOMname properties to CSS2Properties # used for CSSStyleDeclaration to check if allowed properties # but somehow doubled, any better way? CSS2Properties._properties = [] for group in cssutils.profiles.properties: for name in cssutils.profiles.properties[group]: CSS2Properties._properties.append(_toDOMname(name)) # add CSS2Properties to CSSStyleDeclaration: def __named_property_def(DOMname): """ Closure to keep name known in each properties accessor function DOMname is converted to CSSname here, so actual calls use CSSname. """ CSSname = _toCSSname(DOMname) def _get(self): return self._getP(CSSname) def _set(self, value): self._setP(CSSname, value) def _del(self): self._delP(CSSname) return _get, _set, _del # add all CSS2Properties to CSSStyleDeclaration for DOMname in CSS2Properties._properties: setattr(CSS2Properties, DOMname, property(*__named_property_def(DOMname))) cssutils-1.0/src/cssutils/css/csscharsetrule.py0000666000175000017500000001375312126054542020563 0ustar hugohugo"""CSSCharsetRule implements DOM Level 2 CSS CSSCharsetRule.""" __all__ = ['CSSCharsetRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import codecs import cssrule import cssutils import xml.dom class CSSCharsetRule(cssrule.CSSRule): """ The CSSCharsetRule interface represents an @charset rule in a CSS style sheet. The value of the encoding attribute does not affect the encoding of text data in the DOM objects; this encoding is always UTF-16 (also in Python?). After a stylesheet is loaded, the value of the encoding attribute is the value found in the @charset rule. If there was no @charset in the original document, then no CSSCharsetRule is created. The value of the encoding attribute may also be used as a hint for the encoding used on serialization of the style sheet. The value of the @charset rule (and therefore of the CSSCharsetRule) may not correspond to the encoding the document actually came in; character encoding information e.g. in an HTTP header, has priority (see CSS document representation) but this is not reflected in the CSSCharsetRule. This rule is not really needed anymore as setting :attr:`CSSStyleSheet.encoding` is much easier. Format:: charsetrule: CHARSET_SYM S* STRING S* ';' BUT: Only valid format is (single space, double quotes!):: @charset "ENCODING"; """ def __init__(self, encoding=None, parentRule=None, parentStyleSheet=None, readonly=False): """ :param encoding: a valid character encoding :param readonly: defaults to False, not used yet """ super(CSSCharsetRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = '@charset' if encoding: self.encoding = encoding else: self._encoding = None self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(encoding=%r)" % ( self.__class__.__name__, self.encoding) def __str__(self): return u"" % ( self.__class__.__name__, self.encoding, id(self)) def _getCssText(self): """The parsable textual representation.""" return cssutils.ser.do_CSSCharsetRule(self) def _setCssText(self, cssText): """ :param cssText: A parsable DOMString. :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(CSSCharsetRule, self)._setCssText(cssText) wellformed = True tokenizer = self._tokenize2(cssText) if self._type(self._nexttoken(tokenizer)) != self._prods.CHARSET_SYM: wellformed = False self._log.error(u'CSSCharsetRule must start with "@charset "', error=xml.dom.InvalidModificationErr) encodingtoken = self._nexttoken(tokenizer) encodingtype = self._type(encodingtoken) encoding = self._stringtokenvalue(encodingtoken) if self._prods.STRING != encodingtype or not encoding: wellformed = False self._log.error(u'CSSCharsetRule: no encoding found; %r.' % self._valuestr(cssText)) semicolon = self._tokenvalue(self._nexttoken(tokenizer)) EOFtype = self._type(self._nexttoken(tokenizer)) if u';' != semicolon or EOFtype not in ('EOF', None): wellformed = False self._log.error(u'CSSCharsetRule: Syntax Error: %r.' % self._valuestr(cssText)) if wellformed: self.encoding = encoding cssText = property(fget=_getCssText, fset=_setCssText, doc=u"(DOM) The parsable textual representation.") def _setEncoding(self, encoding): """ :param encoding: a valid encoding to be used. Currently only valid Python encodings are allowed. :exceptions: - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this encoding rule is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified encoding value has a syntax error and is unparsable. """ self._checkReadonly() tokenizer = self._tokenize2(encoding) encodingtoken = self._nexttoken(tokenizer) unexpected = self._nexttoken(tokenizer) if not encodingtoken or unexpected or\ self._prods.IDENT != self._type(encodingtoken): self._log.error(u'CSSCharsetRule: Syntax Error in encoding value ' u'%r.' % encoding) else: try: codecs.lookup(encoding) except LookupError: self._log.error(u'CSSCharsetRule: Unknown (Python) encoding %r.' % encoding) else: self._encoding = encoding.lower() encoding = property(lambda self: self._encoding, _setEncoding, doc=u"(DOM)The encoding information used in this @charset rule.") type = property(lambda self: self.CHARSET_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") wellformed = property(lambda self: bool(self.encoding)) cssutils-1.0/src/cssutils/css/cssstyledeclaration.py0000666000175000017500000006505512246442430021612 0ustar hugohugo"""CSSStyleDeclaration implements DOM Level 2 CSS CSSStyleDeclaration and extends CSS2Properties see http://www.w3.org/TR/1998/REC-CSS2-19980512/syndata.html#parsing-errors Unknown properties ------------------ User agents must ignore a declaration with an unknown property. For example, if the style sheet is:: H1 { color: red; rotation: 70minutes } the user agent will treat this as if the style sheet had been:: H1 { color: red } Cssutils gives a message about any unknown properties but keeps any property (if syntactically correct). Illegal values -------------- User agents must ignore a declaration with an illegal value. For example:: IMG { float: left } /* correct CSS2 */ IMG { float: left here } /* "here" is not a value of 'float' */ IMG { background: "red" } /* keywords cannot be quoted in CSS2 */ IMG { border-width: 3 } /* a unit must be specified for length values */ A CSS2 parser would honor the first rule and ignore the rest, as if the style sheet had been:: IMG { float: left } IMG { } IMG { } IMG { } Cssutils again will issue a message (WARNING in this case) about invalid CSS2 property values. TODO: This interface is also used to provide a read-only access to the computed values of an element. See also the ViewCSS interface. - return computed values and not literal values - simplify unit pairs/triples/quadruples 2px 2px 2px 2px -> 2px for border/padding... - normalize compound properties like: background: no-repeat left url() #fff -> background: #fff url() no-repeat left """ __all__ = ['CSSStyleDeclaration', 'Property'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssproperties import CSS2Properties from property import Property import cssutils import xml.dom class CSSStyleDeclaration(CSS2Properties, cssutils.util.Base2): """The CSSStyleDeclaration class represents a single CSS declaration block. This class may be used to determine the style properties currently set in a block or to set style properties explicitly within the block. While an implementation may not recognize all CSS properties within a CSS declaration block, it is expected to provide access to all specified properties in the style sheet through the CSSStyleDeclaration interface. Furthermore, implementations that support a specific level of CSS should correctly handle CSS shorthand properties for that level. For a further discussion of shorthand properties, see the CSS2Properties interface. Additionally the CSS2Properties interface is implemented. $css2propertyname All properties defined in the CSS2Properties class are available as direct properties of CSSStyleDeclaration with their respective DOM name, so e.g. ``fontStyle`` for property 'font-style'. These may be used as:: >>> style = CSSStyleDeclaration(cssText='color: red') >>> style.color = 'green' >>> print style.color green >>> del style.color >>> print style.color Format:: [Property: Value Priority?;]* [Property: Value Priority?]? """ def __init__(self, cssText=u'', parentRule=None, readonly=False, validating=None): """ :param cssText: Shortcut, sets CSSStyleDeclaration.cssText :param parentRule: The CSS rule that contains this declaration block or None if this CSSStyleDeclaration is not attached to a CSSRule. :param readonly: defaults to False :param validating: a flag defining if this sheet should be validated on change. Defaults to None, which means defer to the parent stylesheet. """ super(CSSStyleDeclaration, self).__init__() self._parentRule = parentRule self.validating = validating self.cssText = cssText self._readonly = readonly def __contains__(self, nameOrProperty): """Check if a property (or a property with given name) is in style. :param name: a string or Property, uses normalized name and not literalname """ if isinstance(nameOrProperty, Property): name = nameOrProperty.name else: name = self._normalize(nameOrProperty) return name in self.__nnames() def __iter__(self): """Iterator of set Property objects with different normalized names.""" def properties(): for name in self.__nnames(): yield self.getProperty(name) return properties() def keys(self): """Analoguous to standard dict returns property names which are set in this declaration.""" return list(self.__nnames()) def __getitem__(self, CSSName): """Retrieve the value of property ``CSSName`` from this declaration. ``CSSName`` will be always normalized. """ return self.getPropertyValue(CSSName) def __setitem__(self, CSSName, value): """Set value of property ``CSSName``. ``value`` may also be a tuple of (value, priority), e.g. style['color'] = ('red', 'important') ``CSSName`` will be always normalized. """ priority = None if isinstance(value, tuple): value, priority = value return self.setProperty(CSSName, value, priority) def __delitem__(self, CSSName): """Delete property ``CSSName`` from this declaration. If property is not in this declaration return u'' just like removeProperty. ``CSSName`` will be always normalized. """ return self.removeProperty(CSSName) def __setattr__(self, n, v): """Prevent setting of unknown properties on CSSStyleDeclaration which would not work anyway. For these ``CSSStyleDeclaration.setProperty`` MUST be called explicitly! TODO: implementation of known is not really nice, any alternative? """ known = ['_tokenizer', '_log', '_ttypes', '_seq', 'seq', 'parentRule', '_parentRule', 'cssText', 'valid', 'wellformed', 'validating', '_readonly', '_profiles', '_validating'] known.extend(CSS2Properties._properties) if n in known: super(CSSStyleDeclaration, self).__setattr__(n, v) else: raise AttributeError(u'Unknown CSS Property, ' u'``CSSStyleDeclaration.setProperty("%s", ' u'...)`` MUST be used.' % n) def __repr__(self): return u"cssutils.css.%s(cssText=%r)" % ( self.__class__.__name__, self.getCssText(separator=u' ')) def __str__(self): return u"" % ( self.__class__.__name__, self.length, len(self.getProperties(all=True)), id(self)) def __nnames(self): """Return iterator for all different names in order as set if names are set twice the last one is used (double reverse!) """ names = [] for item in reversed(self.seq): val = item.value if isinstance(val, Property) and not val.name in names: names.append(val.name) return reversed(names) # overwritten accessor functions for CSS2Properties' properties def _getP(self, CSSName): """(DOM CSS2Properties) Overwritten here and effectively the same as ``self.getPropertyValue(CSSname)``. Parameter is in CSSname format ('font-style'), see CSS2Properties. Example:: >>> style = CSSStyleDeclaration(cssText='font-style:italic;') >>> print style.fontStyle italic """ return self.getPropertyValue(CSSName) def _setP(self, CSSName, value): """(DOM CSS2Properties) Overwritten here and effectively the same as ``self.setProperty(CSSname, value)``. Only known CSS2Properties may be set this way, otherwise an AttributeError is raised. For these unknown properties ``setPropertyValue(CSSname, value)`` has to be called explicitly. Also setting the priority of properties needs to be done with a call like ``setPropertyValue(CSSname, value, priority)``. Example:: >>> style = CSSStyleDeclaration() >>> style.fontStyle = 'italic' >>> # or >>> style.setProperty('font-style', 'italic', '!important') """ self.setProperty(CSSName, value) # TODO: Shorthand ones def _delP(self, CSSName): """(cssutils only) Overwritten here and effectively the same as ``self.removeProperty(CSSname)``. Example:: >>> style = CSSStyleDeclaration(cssText='font-style:italic;') >>> del style.fontStyle >>> print style.fontStyle """ self.removeProperty(CSSName) def children(self): """Generator yielding any known child in this declaration including *all* properties, comments or CSSUnknownrules. """ for item in self._seq: yield item.value def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_css_CSSStyleDeclaration(self) def _setCssText(self, cssText): """Setting this attribute will result in the parsing of the new value and resetting of all the properties in the declaration block including the removal or addition of properties. :exceptions: - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this declaration is readonly or a property is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. """ self._checkReadonly() tokenizer = self._tokenize2(cssText) # for closures: must be a mutable new = {'wellformed': True} def ident(expected, seq, token, tokenizer=None): # a property tokens = self._tokensupto2(tokenizer, starttoken=token, semicolon=True) if self._tokenvalue(tokens[-1]) == u';': tokens.pop() property = Property(parent=self) property.cssText = tokens if property.wellformed: seq.append(property, 'Property') else: self._log.error(u'CSSStyleDeclaration: Syntax Error in ' u'Property: %s' % self._valuestr(tokens)) # does not matter in this case return expected def unexpected(expected, seq, token, tokenizer=None): # error, find next ; or } to omit upto next property ignored = self._tokenvalue(token) + self._valuestr( self._tokensupto2(tokenizer, propertyvalueendonly=True)) self._log.error(u'CSSStyleDeclaration: Unexpected token, ignoring ' 'upto %r.' % ignored,token) # does not matter in this case return expected def char(expected, seq, token, tokenizer=None): # a standalone ; or error... if self._tokenvalue(token) == u';': self._log.info(u'CSSStyleDeclaration: Stripped standalone semicolon' u': %s' % self._valuestr([token]), neverraise=True) return expected else: return unexpected(expected, seq, token, tokenizer) # [Property: Value;]* Property: Value? newseq = self._tempSeq() wellformed, expected = self._parse(expected=None, seq=newseq, tokenizer=tokenizer, productions={'IDENT': ident, 'CHAR': char}, default=unexpected) # wellformed set by parse for item in newseq: item.value._parent = self # do not check wellformed as invalid things are removed anyway self._setSeq(newseq) cssText = property(_getCssText, _setCssText, doc=u"(DOM) A parsable textual representation of the " u"declaration block excluding the surrounding curly " u"braces.") def getCssText(self, separator=None): """ :returns: serialized property cssText, each property separated by given `separator` which may e.g. be ``u''`` to be able to use cssText directly in an HTML style attribute. ``;`` is part of each property (except the last one) and **cannot** be set with separator! """ return cssutils.ser.do_css_CSSStyleDeclaration(self, separator) def _setParentRule(self, parentRule): self._parentRule = parentRule # for x in self.children(): # x.parent = self parentRule = property(lambda self: self._parentRule, _setParentRule, doc="(DOM) The CSS rule that contains this declaration block or " "None if this CSSStyleDeclaration is not attached to a CSSRule.") def getProperties(self, name=None, all=False): """ :param name: optional `name` of properties which are requested. Only properties with this **always normalized** `name` are returned. If `name` is ``None`` all properties are returned (at least one for each set name depending on parameter `all`). :param all: if ``False`` (DEFAULT) only the effective properties are returned. If name is given a list with only one property is returned. if ``True`` all properties including properties set multiple times with different values or priorities for different UAs are returned. The order of the properties is fully kept as in the original stylesheet. :returns: a list of :class:`~cssutils.css.Property` objects set in this declaration. """ if name and not all: # single prop but list p = self.getProperty(name) if p: return [p] else: return [] elif not all: # effective Properties in name order return [self.getProperty(name) for name in self.__nnames()] else: # all properties or all with this name nname = self._normalize(name) properties = [] for item in self.seq: val = item.value if isinstance(val, Property) and ( (bool(nname) == False) or (val.name == nname)): properties.append(val) return properties def getProperty(self, name, normalize=True): """ :param name: of the CSS property, always lowercase (even if not normalized) :param normalize: if ``True`` (DEFAULT) name will be normalized (lowercase, no simple escapes) so "color", "COLOR" or "C\olor" will all be equivalent If ``False`` may return **NOT** the effective value but the effective for the unnormalized name. :returns: the effective :class:`~cssutils.css.Property` object. """ nname = self._normalize(name) found = None for item in reversed(self.seq): val = item.value if isinstance(val, Property): if (normalize and nname == val.name) or name == val.literalname: if val.priority: return val elif not found: found = val return found def getPropertyCSSValue(self, name, normalize=True): """ :param name: of the CSS property, always lowercase (even if not normalized) :param normalize: if ``True`` (DEFAULT) name will be normalized (lowercase, no simple escapes) so "color", "COLOR" or "C\olor" will all be equivalent If ``False`` may return **NOT** the effective value but the effective for the unnormalized name. :returns: :class:`~cssutils.css.CSSValue`, the value of the effective property if it has been explicitly set for this declaration block. (DOM) Used to retrieve the object representation of the value of a CSS property if it has been explicitly set within this declaration block. Returns None if the property has not been set. (This method returns None if the property is a shorthand property. Shorthand property values can only be accessed and modified as strings, using the getPropertyValue and setProperty methods.) **cssutils currently always returns a CSSValue if the property is set.** for more on shorthand properties see http://www.dustindiaz.com/css-shorthand/ """ nname = self._normalize(name) if nname in self._SHORTHANDPROPERTIES: self._log.info(u'CSSValue for shorthand property "%s" should be ' u'None, this may be implemented later.' % nname, neverraise=True) p = self.getProperty(name, normalize) if p: return p.propertyValue else: return None def getPropertyValue(self, name, normalize=True): """ :param name: of the CSS property, always lowercase (even if not normalized) :param normalize: if ``True`` (DEFAULT) name will be normalized (lowercase, no simple escapes) so "color", "COLOR" or "C\olor" will all be equivalent If ``False`` may return **NOT** the effective value but the effective for the unnormalized name. :returns: the value of the effective property if it has been explicitly set for this declaration block. Returns the empty string if the property has not been set. """ p = self.getProperty(name, normalize) if p: return p.value else: return u'' def getPropertyPriority(self, name, normalize=True): """ :param name: of the CSS property, always lowercase (even if not normalized) :param normalize: if ``True`` (DEFAULT) name will be normalized (lowercase, no simple escapes) so "color", "COLOR" or "C\olor" will all be equivalent If ``False`` may return **NOT** the effective value but the effective for the unnormalized name. :returns: the priority of the effective CSS property (e.g. the "important" qualifier) if the property has been explicitly set in this declaration block. The empty string if none exists. """ p = self.getProperty(name, normalize) if p: return p.priority else: return u'' def removeProperty(self, name, normalize=True): """ (DOM) Used to remove a CSS property if it has been explicitly set within this declaration block. :param name: of the CSS property :param normalize: if ``True`` (DEFAULT) name will be normalized (lowercase, no simple escapes) so "color", "COLOR" or "C\olor" will all be equivalent. The effective Property value is returned and *all* Properties with ``Property.name == name`` are removed. If ``False`` may return **NOT** the effective value but the effective for the unnormalized `name` only. Also only the Properties with the literal name `name` are removed. :returns: the value of the property if it has been explicitly set for this declaration block. Returns the empty string if the property has not been set or the property name does not correspond to a known CSS property :exceptions: - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this declaration is readonly or the property is readonly. """ self._checkReadonly() r = self.getPropertyValue(name, normalize=normalize) newseq = self._tempSeq() if normalize: # remove all properties with name == nname nname = self._normalize(name) for item in self.seq: if not (isinstance(item.value, Property) and item.value.name == nname): newseq.appendItem(item) else: # remove all properties with literalname == name for item in self.seq: if not (isinstance(item.value, Property) and item.value.literalname == name): newseq.appendItem(item) self._setSeq(newseq) return r def setProperty(self, name, value=None, priority=u'', normalize=True, replace=True): """(DOM) Set a property value and priority within this declaration block. :param name: of the CSS property to set (in W3C DOM the parameter is called "propertyName"), always lowercase (even if not normalized) If a property with this `name` is present it will be reset. cssutils also allowed `name` to be a :class:`~cssutils.css.Property` object, all other parameter are ignored in this case :param value: the new value of the property, ignored if `name` is a Property. :param priority: the optional priority of the property (e.g. "important"), ignored if `name` is a Property. :param normalize: if True (DEFAULT) `name` will be normalized (lowercase, no simple escapes) so "color", "COLOR" or "C\olor" will all be equivalent :param replace: if True (DEFAULT) the given property will replace a present property. If False a new property will be added always. The difference to `normalize` is that two or more properties with the same name may be set, useful for e.g. stuff like:: background: red; background: rgba(255, 0, 0, 0.5); which defines the same property but only capable UAs use the last property value, older ones use the first value. :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this declaration is readonly or the property is readonly. """ self._checkReadonly() if isinstance(name, Property): newp = name name = newp.literalname elif not value: # empty string or None effectively removed property return self.removeProperty(name) else: newp = Property(name, value, priority) if newp.wellformed: if replace: # check if update nname = self._normalize(name) properties = self.getProperties(name, all=(not normalize)) for property in reversed(properties): if normalize and property.name == nname: property.propertyValue = newp.propertyValue.cssText property.priority = newp.priority return elif property.literalname == name: property.propertyValue = newp.propertyValue.cssText property.priority = newp.priority return # not yet set or forced omit replace newp.parent = self self.seq._readonly = False self.seq.append(newp, 'Property') self.seq._readonly = True else: self._log.warn(u'Invalid Property: %s: %s %s' % (name, value, priority)) def item(self, index): """(DOM) Retrieve the properties that have been explicitly set in this declaration block. The order of the properties retrieved using this method does not have to be the order in which they were set. This method can be used to iterate over all properties in this declaration block. :param index: of the property to retrieve, negative values behave like negative indexes on Python lists, so -1 is the last element :returns: the name of the property at this ordinal position. The empty string if no property exists at this position. **ATTENTION:** Only properties with different names are counted. If two properties with the same name are present in this declaration only the effective one is included. :meth:`item` and :attr:`length` work on the same set here. """ names = list(self.__nnames()) try: return names[index] except IndexError: return u'' length = property(lambda self: len(list(self.__nnames())), doc=u"(DOM) The number of distinct properties that have " u"been explicitly in this declaration block. The " u"range of valid indices is 0 to length-1 inclusive. " u"These are properties with a different ``name`` " u"only. :meth:`item` and :attr:`length` work on the " u"same set here.") def _getValidating(self): try: # CSSParser.parseX() sets validating of stylesheet return self.parentRule.parentStyleSheet.validating except AttributeError: # CSSParser.parseStyle() sets validating of declaration if self._validating is not None: return self._validating # default return True def _setValidating(self, validating): self._validating = validating validating = property(_getValidating, _setValidating, doc=u"If ``True`` this declaration validates " u"contained properties. The parent StyleSheet " u"validation setting does *always* win though so " u"even if validating is True it may not validate " u"if the StyleSheet defines else!") cssutils-1.0/src/cssutils/css/cssvariablesrule.py0000666000175000017500000001746212126054542021103 0ustar hugohugo"""CSSVariables implements (and only partly) experimental `CSS Variables `_ """ __all__ = ['CSSVariablesRule'] __docformat__ = 'restructuredtext' __version__ = '$Id: cssfontfacerule.py 1818 2009-07-30 21:39:00Z cthedot $' from cssvariablesdeclaration import CSSVariablesDeclaration import cssrule import cssutils import xml.dom class CSSVariablesRule(cssrule.CSSRule): """ The CSSVariablesRule interface represents a @variables rule within a CSS style sheet. The @variables rule is used to specify variables. cssutils uses a :class:`~cssutils.css.CSSVariablesDeclaration` to represent the variables. Format:: variables VARIABLES_SYM S* medium [ COMMA S* medium ]* LBRACE S* variableset* '}' S* ; for variableset see :class:`cssutils.css.CSSVariablesDeclaration` **Media are not implemented. Reason is that cssutils is using CSS variables in a kind of preprocessing and therefor no media information is available at this stage. For now do not use media!** Example:: @variables { CorporateLogoBGColor: #fe8d12; } div.logoContainer { background-color: var(CorporateLogoBGColor); } """ def __init__(self, mediaText=None, variables=None, parentRule=None, parentStyleSheet=None, readonly=False): """ If readonly allows setting of properties in constructor only. """ super(CSSVariablesRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = u'@variables' # dummy self._media = cssutils.stylesheets.MediaList(mediaText, readonly=readonly) if variables: self.variables = variables else: self.variables = CSSVariablesDeclaration(parentRule=self) self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(mediaText=%r, variables=%r)" % ( self.__class__.__name__, self._media.mediaText, self.variables.cssText) def __str__(self): return u"" % (self.__class__.__name__, self._media.mediaText, self.variables.cssText, self.valid, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSVariablesRule(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. Format:: variables : VARIABLES_SYM S* medium [ COMMA S* medium ]* LBRACE S* variableset* '}' S* ; variableset : LBRACE S* vardeclaration [ ';' S* vardeclaration ]* '}' S* ; """ super(CSSVariablesRule, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) attoken = self._nexttoken(tokenizer, None) if self._type(attoken) != self._prods.VARIABLES_SYM: self._log.error(u'CSSVariablesRule: No CSSVariablesRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: newVariables = CSSVariablesDeclaration(parentRule=self) ok = True beforetokens, brace = self._tokensupto2(tokenizer, blockstartonly=True, separateEnd=True) if self._tokenvalue(brace) != u'{': ok = False self._log.error(u'CSSVariablesRule: No start { of variable ' u'declaration found: %r' % self._valuestr(cssText), brace) # parse stuff before { which should be comments and S only new = {'wellformed': True} newseq = self._tempSeq()#[] beforewellformed, expected = self._parse(expected=':', seq=newseq, tokenizer=self._tokenize2(beforetokens), productions={}) ok = ok and beforewellformed and new['wellformed'] variablestokens, braceorEOFtoken = self._tokensupto2(tokenizer, blockendonly=True, separateEnd=True) val, type_ = self._tokenvalue(braceorEOFtoken), \ self._type(braceorEOFtoken) if val != u'}' and type_ != 'EOF': ok = False self._log.error(u'CSSVariablesRule: No "}" after variables ' u'declaration found: %r' % self._valuestr(cssText)) nonetoken = self._nexttoken(tokenizer) if nonetoken: ok = False self._log.error(u'CSSVariablesRule: Trailing content found.', token=nonetoken) if 'EOF' == type_: # add again as variables needs it variablestokens.append(braceorEOFtoken) # SET but may raise: newVariables.cssText = variablestokens if ok: # contains probably comments only upto { self._setSeq(newseq) self.variables = newVariables cssText = property(_getCssText, _setCssText, doc=u"(DOM) The parsable textual representation of this " u"rule.") media = property(doc=u"NOT IMPLEMENTED! As cssutils resolves variables "\ u"during serializing media information is lost.") def _setVariables(self, variables): """ :param variables: a CSSVariablesDeclaration or string """ self._checkReadonly() if isinstance(variables, basestring): self._variables = CSSVariablesDeclaration(cssText=variables, parentRule=self) else: variables._parentRule = self self._variables = variables variables = property(lambda self: self._variables, _setVariables, doc=u"(DOM) The variables of this rule set, a " u":class:`cssutils.css.CSSVariablesDeclaration`.") type = property(lambda self: self.VARIABLES_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") valid = property(lambda self: True, doc='NOT IMPLEMTED REALLY (TODO)') # constant but needed: wellformed = property(lambda self: True) cssutils-1.0/src/cssutils/css/colors.py0000666000175000017500000001501512126054542017023 0ustar hugohugo# -*- coding: utf-8 -*- """ Built from something like this: print [ ( row[2].text_content().strip(), eval(row[4].text_content().strip()) ) for row in lxml.html.parse('http://www.w3.org/TR/css3-color/') .xpath("//*[@class='colortable']//tr[position()>1]") ] by Simon Sapin """ COLORS = { 'transparent': (0, 0, 0, 0.0), 'black': (0, 0, 0, 1.0), 'silver': (192, 192, 192, 1.0), 'gray': (128, 128, 128, 1.0), 'white': (255, 255, 255, 1.0), 'maroon': (128, 0, 0, 1.0), 'red': (255, 0, 0, 1.0), 'purple': (128, 0, 128, 1.0), 'fuchsia': (255, 0, 255, 1.0), 'green': (0, 128, 0, 1.0), 'lime': (0, 255, 0, 1.0), 'olive': (128, 128, 0, 1.0), 'yellow': (255, 255, 0, 1.0), 'navy': (0, 0, 128, 1.0), 'blue': (0, 0, 255, 1.0), 'teal': (0, 128, 128, 1.0), 'aqua': (0, 255, 255, 1.0), 'aliceblue': (240, 248, 255, 1.0), 'antiquewhite': (250, 235, 215, 1.0), 'aqua': (0, 255, 255, 1.0), 'aquamarine': (127, 255, 212, 1.0), 'azure': (240, 255, 255, 1.0), 'beige': (245, 245, 220, 1.0), 'bisque': (255, 228, 196, 1.0), 'black': (0, 0, 0, 1.0), 'blanchedalmond': (255, 235, 205, 1.0), 'blue': (0, 0, 255, 1.0), 'blueviolet': (138, 43, 226, 1.0), 'brown': (165, 42, 42, 1.0), 'burlywood': (222, 184, 135, 1.0), 'cadetblue': (95, 158, 160, 1.0), 'chartreuse': (127, 255, 0, 1.0), 'chocolate': (210, 105, 30, 1.0), 'coral': (255, 127, 80, 1.0), 'cornflowerblue': (100, 149, 237, 1.0), 'cornsilk': (255, 248, 220, 1.0), 'crimson': (220, 20, 60, 1.0), 'cyan': (0, 255, 255, 1.0), 'darkblue': (0, 0, 139, 1.0), 'darkcyan': (0, 139, 139, 1.0), 'darkgoldenrod': (184, 134, 11, 1.0), 'darkgray': (169, 169, 169, 1.0), 'darkgreen': (0, 100, 0, 1.0), 'darkgrey': (169, 169, 169, 1.0), 'darkkhaki': (189, 183, 107, 1.0), 'darkmagenta': (139, 0, 139, 1.0), 'darkolivegreen': (85, 107, 47, 1.0), 'darkorange': (255, 140, 0, 1.0), 'darkorchid': (153, 50, 204, 1.0), 'darkred': (139, 0, 0, 1.0), 'darksalmon': (233, 150, 122, 1.0), 'darkseagreen': (143, 188, 143, 1.0), 'darkslateblue': (72, 61, 139, 1.0), 'darkslategray': (47, 79, 79, 1.0), 'darkslategrey': (47, 79, 79, 1.0), 'darkturquoise': (0, 206, 209, 1.0), 'darkviolet': (148, 0, 211, 1.0), 'deeppink': (255, 20, 147, 1.0), 'deepskyblue': (0, 191, 255, 1.0), 'dimgray': (105, 105, 105, 1.0), 'dimgrey': (105, 105, 105, 1.0), 'dodgerblue': (30, 144, 255, 1.0), 'firebrick': (178, 34, 34, 1.0), 'floralwhite': (255, 250, 240, 1.0), 'forestgreen': (34, 139, 34, 1.0), 'fuchsia': (255, 0, 255, 1.0), 'gainsboro': (220, 220, 220, 1.0), 'ghostwhite': (248, 248, 255, 1.0), 'gold': (255, 215, 0, 1.0), 'goldenrod': (218, 165, 32, 1.0), 'gray': (128, 128, 128, 1.0), 'green': (0, 128, 0, 1.0), 'greenyellow': (173, 255, 47, 1.0), 'grey': (128, 128, 128, 1.0), 'honeydew': (240, 255, 240, 1.0), 'hotpink': (255, 105, 180, 1.0), 'indianred': (205, 92, 92, 1.0), 'indigo': (75, 0, 130, 1.0), 'ivory': (255, 255, 240, 1.0), 'khaki': (240, 230, 140, 1.0), 'lavender': (230, 230, 250, 1.0), 'lavenderblush': (255, 240, 245, 1.0), 'lawngreen': (124, 252, 0, 1.0), 'lemonchiffon': (255, 250, 205, 1.0), 'lightblue': (173, 216, 230, 1.0), 'lightcoral': (240, 128, 128, 1.0), 'lightcyan': (224, 255, 255, 1.0), 'lightgoldenrodyellow': (250, 250, 210, 1.0), 'lightgray': (211, 211, 211, 1.0), 'lightgreen': (144, 238, 144, 1.0), 'lightgrey': (211, 211, 211, 1.0), 'lightpink': (255, 182, 193, 1.0), 'lightsalmon': (255, 160, 122, 1.0), 'lightseagreen': (32, 178, 170, 1.0), 'lightskyblue': (135, 206, 250, 1.0), 'lightslategray': (119, 136, 153, 1.0), 'lightslategrey': (119, 136, 153, 1.0), 'lightsteelblue': (176, 196, 222, 1.0), 'lightyellow': (255, 255, 224, 1.0), 'lime': (0, 255, 0, 1.0), 'limegreen': (50, 205, 50, 1.0), 'linen': (250, 240, 230, 1.0), 'magenta': (255, 0, 255, 1.0), 'maroon': (128, 0, 0, 1.0), 'mediumaquamarine': (102, 205, 170, 1.0), 'mediumblue': (0, 0, 205, 1.0), 'mediumorchid': (186, 85, 211, 1.0), 'mediumpurple': (147, 112, 219, 1.0), 'mediumseagreen': (60, 179, 113, 1.0), 'mediumslateblue': (123, 104, 238, 1.0), 'mediumspringgreen': (0, 250, 154, 1.0), 'mediumturquoise': (72, 209, 204, 1.0), 'mediumvioletred': (199, 21, 133, 1.0), 'midnightblue': (25, 25, 112, 1.0), 'mintcream': (245, 255, 250, 1.0), 'mistyrose': (255, 228, 225, 1.0), 'moccasin': (255, 228, 181, 1.0), 'navajowhite': (255, 222, 173, 1.0), 'navy': (0, 0, 128, 1.0), 'oldlace': (253, 245, 230, 1.0), 'olive': (128, 128, 0, 1.0), 'olivedrab': (107, 142, 35, 1.0), 'orange': (255, 165, 0, 1.0), 'orangered': (255, 69, 0, 1.0), 'orchid': (218, 112, 214, 1.0), 'palegoldenrod': (238, 232, 170, 1.0), 'palegreen': (152, 251, 152, 1.0), 'paleturquoise': (175, 238, 238, 1.0), 'palevioletred': (219, 112, 147, 1.0), 'papayawhip': (255, 239, 213, 1.0), 'peachpuff': (255, 218, 185, 1.0), 'peru': (205, 133, 63, 1.0), 'pink': (255, 192, 203, 1.0), 'plum': (221, 160, 221, 1.0), 'powderblue': (176, 224, 230, 1.0), 'purple': (128, 0, 128, 1.0), 'red': (255, 0, 0, 1.0), 'rosybrown': (188, 143, 143, 1.0), 'royalblue': (65, 105, 225, 1.0), 'saddlebrown': (139, 69, 19, 1.0), 'salmon': (250, 128, 114, 1.0), 'sandybrown': (244, 164, 96, 1.0), 'seagreen': (46, 139, 87, 1.0), 'seashell': (255, 245, 238, 1.0), 'sienna': (160, 82, 45, 1.0), 'silver': (192, 192, 192, 1.0), 'skyblue': (135, 206, 235, 1.0), 'slateblue': (106, 90, 205, 1.0), 'slategray': (112, 128, 144, 1.0), 'slategrey': (112, 128, 144, 1.0), 'snow': (255, 250, 250, 1.0), 'springgreen': (0, 255, 127, 1.0), 'steelblue': (70, 130, 180, 1.0), 'tan': (210, 180, 140, 1.0), 'teal': (0, 128, 128, 1.0), 'thistle': (216, 191, 216, 1.0), 'tomato': (255, 99, 71, 1.0), 'turquoise': (64, 224, 208, 1.0), 'violet': (238, 130, 238, 1.0), 'wheat': (245, 222, 179, 1.0), 'white': (255, 255, 255, 1.0), 'whitesmoke': (245, 245, 245, 1.0), 'yellow': (255, 255, 0, 1.0), 'yellowgreen': (154, 205, 50, 1.0), } cssutils-1.0/src/cssutils/css/cssrulelist.py0000666000175000017500000000376712126054542020111 0ustar hugohugo"""CSSRuleList implements DOM Level 2 CSS CSSRuleList. Partly also http://dev.w3.org/csswg/cssom/#the-cssrulelist.""" __all__ = ['CSSRuleList'] __docformat__ = 'restructuredtext' __version__ = '$Id$' class CSSRuleList(list): """The CSSRuleList object represents an (ordered) list of statements. The items in the CSSRuleList are accessible via an integral index, starting from 0. Subclasses a standard Python list so theoretically all standard list methods are available. Setting methods like ``__init__``, ``append``, ``extend`` or ``__setslice__`` are added later on instances of this class if so desired. E.g. CSSStyleSheet adds ``append`` which is not available in a simple instance of this class! """ def __init__(self, *ignored): "Nothing is set as this must also be defined later." pass def __notimplemented(self, *ignored): "Implemented in class using a CSSRuleList only." raise NotImplementedError( 'Must be implemented by class using an instance of this class.') append = extend = __setitem__ = __setslice__ = __notimplemented def item(self, index): """(DOM) Retrieve a CSS rule by ordinal `index`. The order in this collection represents the order of the rules in the CSS style sheet. If index is greater than or equal to the number of rules in the list, this returns None. Returns CSSRule, the style rule at the index position in the CSSRuleList, or None if that is not a valid index. """ try: return self[index] except IndexError: return None length = property(lambda self: len(self), doc=u"(DOM) The number of CSSRules in the list.") def rulesOfType(self, type): """Yield the rules which have the given `type` only, one of the constants defined in :class:`cssutils.css.CSSRule`.""" for r in self: if r.type == type: yield r cssutils-1.0/src/cssutils/css/cssstylerule.py0000666000175000017500000002200012126054542020253 0ustar hugohugo"""CSSStyleRule implements DOM Level 2 CSS CSSStyleRule.""" __all__ = ['CSSStyleRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssstyledeclaration import CSSStyleDeclaration from selectorlist import SelectorList import cssrule import cssutils import xml.dom class CSSStyleRule(cssrule.CSSRule): """The CSSStyleRule object represents a ruleset specified (if any) in a CSS style sheet. It provides access to a declaration block as well as to the associated group of selectors. Format:: : selector [ COMMA S* selector ]* LBRACE S* declaration [ ';' S* declaration ]* '}' S* ; """ def __init__(self, selectorText=None, style=None, parentRule=None, parentStyleSheet=None, readonly=False): """ :Parameters: selectorText string parsed into selectorList style string parsed into CSSStyleDeclaration for this CSSStyleRule readonly if True allows setting of properties in constructor only """ super(CSSStyleRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self.selectorList = SelectorList() if selectorText: self.selectorText = selectorText if style: self.style = style else: self.style = CSSStyleDeclaration() self._readonly = readonly def __repr__(self): if self._namespaces: st = (self.selectorText, self._namespaces) else: st = self.selectorText return u"cssutils.css.%s(selectorText=%r, style=%r)" % ( self.__class__.__name__, st, self.style.cssText) def __str__(self): return u"" % (self.__class__.__name__, self.selectorText, self.style.cssText, self._namespaces, id(self)) def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSStyleRule(self) def _setCssText(self, cssText): """ :param cssText: a parseable string or a tuple of (cssText, dict-of-namespaces) :exceptions: - :exc:`~xml.dom.NamespaceErr`: Raised if the specified selector uses an unknown namespace prefix. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. """ super(CSSStyleRule, self)._setCssText(cssText) # might be (cssText, namespaces) cssText, namespaces = self._splitNamespacesOff(cssText) try: # use parent style sheet ones if available namespaces = self.parentStyleSheet.namespaces except AttributeError: pass tokenizer = self._tokenize2(cssText) selectortokens = self._tokensupto2(tokenizer, blockstartonly=True) styletokens = self._tokensupto2(tokenizer, blockendonly=True) trail = self._nexttoken(tokenizer) if trail: self._log.error(u'CSSStyleRule: Trailing content: %s' % self._valuestr(cssText), token=trail) elif not selectortokens: self._log.error(u'CSSStyleRule: No selector found: %r' % self._valuestr(cssText)) elif self._tokenvalue(selectortokens[0]).startswith(u'@'): self._log.error(u'CSSStyleRule: No style rule: %r' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: newSelectorList = SelectorList(parentRule=self) newStyle = CSSStyleDeclaration(parentRule=self) ok = True bracetoken = selectortokens.pop() if self._tokenvalue(bracetoken) != u'{': ok = False self._log.error( u'CSSStyleRule: No start { of style declaration found: %r' % self._valuestr(cssText), bracetoken) elif not selectortokens: ok = False self._log.error(u'CSSStyleRule: No selector found: %r.' % self._valuestr(cssText), bracetoken) # SET newSelectorList.selectorText = (selectortokens, namespaces) if not styletokens: ok = False self._log.error( u'CSSStyleRule: No style declaration or "}" found: %r' % self._valuestr(cssText)) else: braceorEOFtoken = styletokens.pop() val, typ = self._tokenvalue(braceorEOFtoken),\ self._type(braceorEOFtoken) if val != u'}' and typ != 'EOF': ok = False self._log.error(u'CSSStyleRule: No "}" after style ' u'declaration found: %r' % self._valuestr(cssText)) else: if 'EOF' == typ: # add again as style needs it styletokens.append(braceorEOFtoken) # SET, may raise: newStyle.cssText = styletokens if ok: self.selectorList = newSelectorList self.style = newStyle cssText = property(_getCssText, _setCssText, doc=u"(DOM) The parsable textual representation of this " u"rule.") def __getNamespaces(self): """Uses children namespaces if not attached to a sheet, else the sheet's ones.""" try: return self.parentStyleSheet.namespaces except AttributeError: return self.selectorList._namespaces _namespaces = property(__getNamespaces, doc=u"If this Rule is attached to a CSSStyleSheet " u"the namespaces of that sheet are mirrored " u"here. While the Rule is not attached the " u"namespaces of selectorList are used.""") def _setSelectorList(self, selectorList): """ :param selectorList: A SelectorList which replaces the current selectorList object """ self._checkReadonly() selectorList._parentRule = self self._selectorList = selectorList _selectorList = None selectorList = property(lambda self: self._selectorList, _setSelectorList, doc=u"The SelectorList of this rule.") def _setSelectorText(self, selectorText): """ wrapper for cssutils SelectorList object :param selectorText: of type string, might also be a comma separated list of selectors :exceptions: - :exc:`~xml.dom.NamespaceErr`: Raised if the specified selector uses an unknown namespace prefix. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. """ self._checkReadonly() sl = SelectorList(selectorText=selectorText, parentRule=self) if sl.wellformed: self._selectorList = sl selectorText = property(lambda self: self._selectorList.selectorText, _setSelectorText, doc=u"(DOM) The textual representation of the " u"selector for the rule set.") def _setStyle(self, style): """ :param style: A string or CSSStyleDeclaration which replaces the current style object. """ self._checkReadonly() if isinstance(style, basestring): self._style = CSSStyleDeclaration(cssText=style, parentRule=self) else: style._parentRule = self self._style = style style = property(lambda self: self._style, _setStyle, doc=u"(DOM) The declaration-block of this rule set.") type = property(lambda self: self.STYLE_RULE, doc=u"The type of this rule, as defined by a CSSRule " "type constant.") wellformed = property(lambda self: self.selectorList.wellformed) cssutils-1.0/src/cssutils/css/selectorlist.py0000666000175000017500000002076612126054542020247 0ustar hugohugo"""SelectorList is a list of CSS Selector objects. TODO - remove duplicate Selectors. -> CSSOM canonicalize - ??? CSS2 gives a special meaning to the comma (,) in selectors. However, since it is not known if the comma may acquire other meanings in future versions of CSS, the whole statement should be ignored if there is an error anywhere in the selector, even though the rest of the selector may look reasonable in CSS2. Illegal example(s): For example, since the "&" is not a valid token in a CSS2 selector, a CSS2 user agent must ignore the whole second line, and not set the color of H3 to red: """ __all__ = ['SelectorList'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from selector import Selector import cssutils import xml.dom class SelectorList(cssutils.util.Base, cssutils.util.ListSeq): """A list of :class:`~cssutils.css.Selector` objects of a :class:`~cssutils.css.CSSStyleRule`.""" def __init__(self, selectorText=None, parentRule=None, readonly=False): """ :Parameters: selectorText parsable list of Selectors parentRule the parent CSSRule if available """ super(SelectorList, self).__init__() self._parentRule = parentRule if selectorText: self.selectorText = selectorText self._readonly = readonly def __repr__(self): if self._namespaces: st = (self.selectorText, self._namespaces) else: st = self.selectorText return u"cssutils.css.%s(selectorText=%r)" % (self.__class__.__name__, st) def __str__(self): return u"" % (self.__class__.__name__, self.selectorText, self._namespaces, id(self)) def __setitem__(self, index, newSelector): """Overwrite ListSeq.__setitem__ Any duplicate Selectors are **not** removed. """ newSelector = self.__prepareset(newSelector) if newSelector: self.seq[index] = newSelector def __prepareset(self, newSelector, namespaces=None): "Used by appendSelector and __setitem__" if not namespaces: namespaces = {} self._checkReadonly() if not isinstance(newSelector, Selector): newSelector = Selector((newSelector, namespaces), parent=self) if newSelector.wellformed: newSelector._parent = self # maybe set twice but must be! return newSelector def __getNamespaces(self): """Use children namespaces if not attached to a sheet, else the sheet's ones. """ try: return self.parentRule.parentStyleSheet.namespaces except AttributeError: namespaces = {} for selector in self.seq: namespaces.update(selector._namespaces) return namespaces def _getUsedUris(self): "Used by CSSStyleSheet to check if @namespace rules are needed" uris = set() for s in self: uris.update(s._getUsedUris()) return uris _namespaces = property(__getNamespaces, doc="""If this SelectorList is attached to a CSSStyleSheet the namespaces of that sheet are mirrored here. While the SelectorList (or parentRule(s) are not attached the namespaces of all children Selectors are used.""") def append(self, newSelector): "Same as :meth:`appendSelector`." self.appendSelector(newSelector) def appendSelector(self, newSelector): """ Append `newSelector` to this list (a string will be converted to a :class:`~cssutils.css.Selector`). :param newSelector: comma-separated list of selectors (as a single string) or a tuple of `(newSelector, dict-of-namespaces)` :returns: New :class:`~cssutils.css.Selector` or ``None`` if `newSelector` is not wellformed. :exceptions: - :exc:`~xml.dom.NamespaceErr`: Raised if the specified selector uses an unknown namespace prefix. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. """ self._checkReadonly() # might be (selectorText, namespaces) newSelector, namespaces = self._splitNamespacesOff(newSelector) try: # use parent's only if available namespaces = self.parentRule.parentStyleSheet.namespaces except AttributeError: # use already present namespaces plus new given ones _namespaces = self._namespaces _namespaces.update(namespaces) namespaces = _namespaces newSelector = self.__prepareset(newSelector, namespaces) if newSelector: seq = self.seq[:] del self.seq[:] for s in seq: if s.selectorText != newSelector.selectorText: self.seq.append(s) self.seq.append(newSelector) return newSelector def _getSelectorText(self): "Return serialized format." return cssutils.ser.do_css_SelectorList(self) def _setSelectorText(self, selectorText): """ :param selectorText: comma-separated list of selectors or a tuple of (selectorText, dict-of-namespaces) :exceptions: - :exc:`~xml.dom.NamespaceErr`: Raised if the specified selector uses an unknown namespace prefix. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this rule is readonly. """ self._checkReadonly() # might be (selectorText, namespaces) selectorText, namespaces = self._splitNamespacesOff(selectorText) try: # use parent's only if available namespaces = self.parentRule.parentStyleSheet.namespaces except AttributeError: pass wellformed = True tokenizer = self._tokenize2(selectorText) newseq = [] expected = True while True: # find all upto and including next ",", EOF or nothing selectortokens = self._tokensupto2(tokenizer, listseponly=True) if selectortokens: if self._tokenvalue(selectortokens[-1]) == ',': expected = selectortokens.pop() else: expected = None selector = Selector((selectortokens, namespaces), parent=self) if selector.wellformed: newseq.append(selector) else: wellformed = False self._log.error(u'SelectorList: Invalid Selector: %s' % self._valuestr(selectortokens)) else: break # post condition if u',' == expected: wellformed = False self._log.error(u'SelectorList: Cannot end with ",": %r' % self._valuestr(selectorText)) elif expected: wellformed = False self._log.error(u'SelectorList: Unknown Syntax: %r' % self._valuestr(selectorText)) if wellformed: self.seq = newseq selectorText = property(_getSelectorText, _setSelectorText, doc=u"(cssutils) The textual representation of the " u"selector for a rule set.") length = property(lambda self: len(self), doc=u"The number of :class:`~cssutils.css.Selector` " u"objects in the list.") parentRule = property(lambda self: self._parentRule, doc=u"(DOM) The CSS rule that contains this " u"SelectorList or ``None`` if this SelectorList " u"is not attached to a CSSRule.") wellformed = property(lambda self: bool(len(self.seq))) cssutils-1.0/src/cssutils/css/cssvalue.py0000666000175000017500000014566512244401752017366 0ustar hugohugo"""CSSValue related classes - CSSValue implements DOM Level 2 CSS CSSValue - CSSPrimitiveValue implements DOM Level 2 CSS CSSPrimitiveValue - CSSValueList implements DOM Level 2 CSS CSSValueList """ __all__ = ['CSSValue', 'CSSPrimitiveValue', 'CSSValueList', 'RGBColor', 'CSSVariable'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.prodparser import * import cssutils import cssutils.helper import math import re import xml.dom class CSSValue(cssutils.util._NewBase): """The CSSValue interface represents a simple or a complex value. A CSSValue object only occurs in a context of a CSS property. """ # The value is inherited and the cssText contains "inherit". CSS_INHERIT = 0 # The value is a CSSPrimitiveValue. CSS_PRIMITIVE_VALUE = 1 # The value is a CSSValueList. CSS_VALUE_LIST = 2 # The value is a custom value. CSS_CUSTOM = 3 # The value is a CSSVariable. CSS_VARIABLE = 4 _typestrings = {0: 'CSS_INHERIT' , 1: 'CSS_PRIMITIVE_VALUE', 2: 'CSS_VALUE_LIST', 3: 'CSS_CUSTOM', 4: 'CSS_VARIABLE'} def __init__(self, cssText=None, parent=None, readonly=False): """ :param cssText: the parsable cssText of the value :param readonly: defaults to False """ super(CSSValue, self).__init__() self._cssValueType = None self.wellformed = False self.parent = parent if cssText is not None: # may be 0 if isinstance(cssText, int): cssText = unicode(cssText) # if it is an integer elif isinstance(cssText, float): cssText = u'%f' % cssText # if it is a floating point number self.cssText = cssText self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(%r)" % (self.__class__.__name__, self.cssText) def __str__(self): return u"" % (self.__class__.__name__, self.cssValueTypeString, self.cssText, id(self)) def _setCssText(self, cssText): """ Format:: unary_operator : '-' | '+' ; operator : '/' S* | ',' S* | /* empty */ ; expr : term [ operator term ]* ; term : unary_operator? [ NUMBER S* | PERCENTAGE S* | LENGTH S* | EMS S* | EXS S* | ANGLE S* | TIME S* | FREQ S* ] | STRING S* | IDENT S* | URI S* | hexcolor | function | UNICODE-RANGE S* ; function : FUNCTION S* expr ')' S* ; /* * There is a constraint on the color that it must * have either 3 or 6 hex-digits (i.e., [0-9a-fA-F]) * after the "#"; e.g., "#000" is OK, but "#abcd" is not. */ hexcolor : HASH S* ; :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error (according to the attached property) or is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: TODO: Raised if the specified CSS string value represents a different type of values than the values allowed by the CSS property. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this value is readonly. """ self._checkReadonly() # used as operator is , / or S nextSor = u',/' term = Choice(Sequence(PreDef.unary(), Choice(PreDef.number(nextSor=nextSor), PreDef.percentage(nextSor=nextSor), PreDef.dimension(nextSor=nextSor))), PreDef.string(nextSor=nextSor), PreDef.ident(nextSor=nextSor), PreDef.uri(nextSor=nextSor), PreDef.hexcolor(nextSor=nextSor), PreDef.unicode_range(nextSor=nextSor), # special case IE only expression Prod(name='expression', match=lambda t, v: t == self._prods.FUNCTION and ( cssutils.helper.normalize(v) in (u'expression(', u'alpha(', u'blur(', u'chroma(', u'dropshadow(', u'fliph(', u'flipv(', u'glow(', u'gray(', u'invert(', u'mask(', u'shadow(', u'wave(', u'xray(') or v.startswith(u'progid:DXImageTransform.Microsoft.') ), nextSor=nextSor, toSeq=lambda t, tokens: (ExpressionValue._functionName, ExpressionValue( cssutils.helper.pushtoken(t, tokens), parent=self) ) ), # CSS Variable var( PreDef.variable(nextSor=nextSor, toSeq=lambda t, tokens: ('CSSVariable', CSSVariable( cssutils.helper.pushtoken(t, tokens), parent=self) ) ), # calc( PreDef.calc(nextSor=nextSor, toSeq=lambda t, tokens: (CalcValue._functionName, CalcValue( cssutils.helper.pushtoken(t, tokens), parent=self) ) ), # TODO: # # rgb/rgba( # Prod(name='RGBColor', # match=lambda t, v: t == self._prods.FUNCTION and ( # cssutils.helper.normalize(v) in (u'rgb(', # u'rgba(' # ) # ), # nextSor=nextSor, # toSeq=lambda t, tokens: (RGBColor._functionName, # RGBColor( # cssutils.helper.pushtoken(t, tokens), # parent=self) # ) # ), # other functions like rgb( etc PreDef.function(nextSor=nextSor, toSeq=lambda t, tokens: ('FUNCTION', CSSFunction( cssutils.helper.pushtoken(t, tokens), parent=self) ) ) ) operator = Choice(PreDef.S(), PreDef.char('comma', ',', toSeq=lambda t, tokens: ('operator', t[1])), PreDef.char('slash', '/', toSeq=lambda t, tokens: ('operator', t[1])), optional=True) # CSSValue PRODUCTIONS valueprods = Sequence(term, Sequence(operator, # mayEnd this Sequence if whitespace # TODO: only when setting via other class # used by variabledeclaration currently PreDef.char('END', ';', stopAndKeep=True, optional=True), term, minmax=lambda: (0, None))) # parse wellformed, seq, store, notused = ProdParser().parse(cssText, u'CSSValue', valueprods, keepS=True) if wellformed: # - count actual values and set firstvalue which is used later on # - combine comma separated list, e.g. font-family to a single item # - remove S which should be an operator but is no needed count, firstvalue = 0, () newseq = self._tempSeq() i, end = 0, len(seq) while i < end: item = seq[i] if item.type == self._prods.S: pass elif (item.value, item.type) == (u',', 'operator'): # , separared counts as a single STRING for now # URI or STRING value might be a single CHAR too! newseq.appendItem(item) count -= 1 if firstvalue: # list of IDENTs is handled as STRING! if firstvalue[1] == self._prods.IDENT: firstvalue = firstvalue[0], 'STRING' elif item.value == u'/': # / separated items count as one newseq.appendItem(item) elif item.value == u'-' or item.value == u'+': # combine +- and following number or other i += 1 try: next = seq[i] except IndexError: firstvalue = () # raised later break newval = item.value + next.value newseq.append(newval, next.type, item.line, item.col) if not firstvalue: firstvalue = (newval, next.type) count += 1 elif item.type != cssutils.css.CSSComment: newseq.appendItem(item) if not firstvalue: firstvalue = (item.value, item.type) count += 1 else: newseq.appendItem(item) i += 1 if not firstvalue: self._log.error( u'CSSValue: Unknown syntax or no value: %r.' % self._valuestr(cssText)) else: # ok and set self._setSeq(newseq) self.wellformed = wellformed if hasattr(self, '_value'): # only in case of CSSPrimitiveValue, else remove! del self._value if count == 1: # inherit, primitive or variable if isinstance(firstvalue[0], basestring) and\ u'inherit' == cssutils.helper.normalize(firstvalue[0]): self.__class__ = CSSValue self._cssValueType = CSSValue.CSS_INHERIT elif 'CSSVariable' == firstvalue[1]: self.__class__ = CSSVariable self._value = firstvalue # TODO: remove major hack! self._name = firstvalue[0]._name else: self.__class__ = CSSPrimitiveValue self._value = firstvalue elif count > 1: # valuelist self.__class__ = CSSValueList # change items in list to specific type (primitive etc) newseq = self._tempSeq() commalist = [] nexttocommalist = False def itemValue(item): "Reserialized simple item.value" if self._prods.STRING == item.type: return cssutils.helper.string(item.value) elif self._prods.URI == item.type: return cssutils.helper.uri(item.value) elif self._prods.FUNCTION == item.type or\ 'CSSVariable' == item.type: return item.value.cssText else: return item.value def saveifcommalist(commalist, newseq): """ saves items in commalist to seq and items if anything in there """ if commalist: newseq.replace(-1, CSSPrimitiveValue(cssText=u''.join( commalist)), CSSPrimitiveValue, newseq[-1].line, newseq[-1].col) del commalist[:] for i, item in enumerate(self._seq): if issubclass(type(item.value), CSSValue): # set parent of CSSValueList items to the lists # parent item.value.parent = self.parent if item.type in (self._prods.DIMENSION, self._prods.FUNCTION, self._prods.HASH, self._prods.IDENT, self._prods.NUMBER, self._prods.PERCENTAGE, self._prods.STRING, self._prods.URI, self._prods.UNICODE_RANGE, 'CSSVariable'): if nexttocommalist: # wait until complete commalist.append(itemValue(item)) else: saveifcommalist(commalist, newseq) # append new item if hasattr(item.value, 'cssText'): newseq.append(item.value, item.value.__class__, item.line, item.col) else: newseq.append(CSSPrimitiveValue( itemValue(item)), CSSPrimitiveValue, item.line, item.col) nexttocommalist = False elif u',' == item.value: if not commalist: # save last item to commalist commalist.append(itemValue(self._seq[i - 1])) commalist.append(u',') nexttocommalist = True else: if nexttocommalist: commalist.append(item.value.cssText) else: newseq.appendItem(item) saveifcommalist(commalist, newseq) self._setSeq(newseq) else: # should not happen... self.__class__ = CSSValue self._cssValueType = CSSValue.CSS_CUSTOM cssText = property(lambda self: cssutils.ser.do_css_CSSValue(self), _setCssText, doc="A string representation of the current value.") cssValueType = property(lambda self: self._cssValueType, doc="A (readonly) code defining the type of the value.") cssValueTypeString = property( lambda self: CSSValue._typestrings.get(self.cssValueType, None), doc="(readonly) Name of cssValueType.") class CSSPrimitiveValue(CSSValue): """Represents a single CSS Value. May be used to determine the value of a specific style property currently set in a block or to set a specific style property explicitly within the block. Might be obtained from the getPropertyCSSValue method of CSSStyleDeclaration. Conversions are allowed between absolute values (from millimeters to centimeters, from degrees to radians, and so on) but not between relative values. (For example, a pixel value cannot be converted to a centimeter value.) Percentage values can't be converted since they are relative to the parent value (or another property value). There is one exception for color percentage values: since a color percentage value is relative to the range 0-255, a color percentage value can be converted to a number; (see also the RGBColor interface). """ # constant: type of this CSSValue class cssValueType = CSSValue.CSS_PRIMITIVE_VALUE __types = cssutils.cssproductions.CSSProductions # An integer indicating which type of unit applies to the value. CSS_UNKNOWN = 0 # only obtainable via cssText CSS_NUMBER = 1 CSS_PERCENTAGE = 2 CSS_EMS = 3 CSS_EXS = 4 CSS_PX = 5 CSS_CM = 6 CSS_MM = 7 CSS_IN = 8 CSS_PT = 9 CSS_PC = 10 CSS_DEG = 11 CSS_RAD = 12 CSS_GRAD = 13 CSS_MS = 14 CSS_S = 15 CSS_HZ = 16 CSS_KHZ = 17 CSS_DIMENSION = 18 CSS_STRING = 19 CSS_URI = 20 CSS_IDENT = 21 CSS_ATTR = 22 CSS_COUNTER = 23 CSS_RECT = 24 CSS_RGBCOLOR = 25 # NOT OFFICIAL: CSS_RGBACOLOR = 26 CSS_UNICODE_RANGE = 27 _floattypes = (CSS_NUMBER, CSS_PERCENTAGE, CSS_EMS, CSS_EXS, CSS_PX, CSS_CM, CSS_MM, CSS_IN, CSS_PT, CSS_PC, CSS_DEG, CSS_RAD, CSS_GRAD, CSS_MS, CSS_S, CSS_HZ, CSS_KHZ, CSS_DIMENSION) _stringtypes = (CSS_ATTR, CSS_IDENT, CSS_STRING, CSS_URI) _countertypes = (CSS_COUNTER,) _recttypes = (CSS_RECT,) _rbgtypes = (CSS_RGBCOLOR, CSS_RGBACOLOR) _lengthtypes = (CSS_NUMBER, CSS_EMS, CSS_EXS, CSS_PX, CSS_CM, CSS_MM, CSS_IN, CSS_PT, CSS_PC) # oldtype: newType: converterfunc _converter = { # cm <-> mm <-> in, 1 inch is equal to 2.54 centimeters. # pt <-> pc, the points used by CSS 2.1 are equal to 1/72nd of an inch. # pc: picas - 1 pica is equal to 12 points (CSS_CM, CSS_MM): lambda x: x * 10, (CSS_MM, CSS_CM): lambda x: x / 10, (CSS_PT, CSS_PC): lambda x: x * 12, (CSS_PC, CSS_PT): lambda x: x / 12, (CSS_CM, CSS_IN): lambda x: x / 2.54, (CSS_IN, CSS_CM): lambda x: x * 2.54, (CSS_MM, CSS_IN): lambda x: x / 25.4, (CSS_IN, CSS_MM): lambda x: x * 25.4, (CSS_IN, CSS_PT): lambda x: x / 72, (CSS_PT, CSS_IN): lambda x: x * 72, (CSS_CM, CSS_PT): lambda x: x / 2.54 / 72, (CSS_PT, CSS_CM): lambda x: x * 72 * 2.54, (CSS_MM, CSS_PT): lambda x: x / 25.4 / 72, (CSS_PT, CSS_MM): lambda x: x * 72 * 25.4, (CSS_IN, CSS_PC): lambda x: x / 72 / 12, (CSS_PC, CSS_IN): lambda x: x * 12 * 72, (CSS_CM, CSS_PC): lambda x: x / 2.54 / 72 / 12, (CSS_PC, CSS_CM): lambda x: x * 12 * 72 * 2.54, (CSS_MM, CSS_PC): lambda x: x / 25.4 / 72 / 12, (CSS_PC, CSS_MM): lambda x: x * 12 * 72 * 25.4, # hz <-> khz (CSS_KHZ, CSS_HZ): lambda x: x * 1000, (CSS_HZ, CSS_KHZ): lambda x: x / 1000, # s <-> ms (CSS_S, CSS_MS): lambda x: x * 1000, (CSS_MS, CSS_S): lambda x: x / 1000, (CSS_RAD, CSS_DEG): lambda x: math.degrees(x), (CSS_DEG, CSS_RAD): lambda x: math.radians(x), # TODO: convert grad <-> deg or rad #(CSS_RAD, CSS_GRAD): lambda x: math.degrees(x), #(CSS_DEG, CSS_GRAD): lambda x: math.radians(x), #(CSS_GRAD, CSS_RAD): lambda x: math.radians(x), #(CSS_GRAD, CSS_DEG): lambda x: math.radians(x) } def __init__(self, cssText=None, parent=None, readonly=False): """See CSSPrimitiveValue.__init__()""" super(CSSPrimitiveValue, self).__init__(cssText=cssText, parent=parent, readonly=readonly) def __str__(self): return u""\ % (self.__class__.__name__, self.primitiveTypeString, self.cssText, id(self)) _unitnames = ['CSS_UNKNOWN', 'CSS_NUMBER', 'CSS_PERCENTAGE', 'CSS_EMS', 'CSS_EXS', 'CSS_PX', 'CSS_CM', 'CSS_MM', 'CSS_IN', 'CSS_PT', 'CSS_PC', 'CSS_DEG', 'CSS_RAD', 'CSS_GRAD', 'CSS_MS', 'CSS_S', 'CSS_HZ', 'CSS_KHZ', 'CSS_DIMENSION', 'CSS_STRING', 'CSS_URI', 'CSS_IDENT', 'CSS_ATTR', 'CSS_COUNTER', 'CSS_RECT', 'CSS_RGBCOLOR', 'CSS_RGBACOLOR', 'CSS_UNICODE_RANGE' ] _reNumDim = re.compile(ur'([+-]?\d*\.\d+|[+-]?\d+)(.*)$', re.I | re.U | re.X) def _unitDIMENSION(value): """Check val for dimension name.""" units = {'em': 'CSS_EMS', 'ex': 'CSS_EXS', 'px': 'CSS_PX', 'cm': 'CSS_CM', 'mm': 'CSS_MM', 'in': 'CSS_IN', 'pt': 'CSS_PT', 'pc': 'CSS_PC', 'deg': 'CSS_DEG', 'rad': 'CSS_RAD', 'grad': 'CSS_GRAD', 'ms': 'CSS_MS', 's': 'CSS_S', 'hz': 'CSS_HZ', 'khz': 'CSS_KHZ' } val, dim = CSSPrimitiveValue._reNumDim.findall(cssutils.helper.normalize(value))[0] return units.get(dim, 'CSS_DIMENSION') def _unitFUNCTION(value): """Check val for function name.""" units = {'attr(': 'CSS_ATTR', 'counter(': 'CSS_COUNTER', 'rect(': 'CSS_RECT', 'rgb(': 'CSS_RGBCOLOR', 'rgba(': 'CSS_RGBACOLOR', } return units.get(re.findall(ur'^(.*?\()', cssutils.helper.normalize(value.cssText), re.U)[0], 'CSS_UNKNOWN') __unitbytype = { __types.NUMBER: 'CSS_NUMBER', __types.PERCENTAGE: 'CSS_PERCENTAGE', __types.STRING: 'CSS_STRING', __types.UNICODE_RANGE: 'CSS_UNICODE_RANGE', __types.URI: 'CSS_URI', __types.IDENT: 'CSS_IDENT', __types.HASH: 'CSS_RGBCOLOR', __types.DIMENSION: _unitDIMENSION, __types.FUNCTION: _unitFUNCTION } def __set_primitiveType(self): """primitiveType is readonly but is set lazy if accessed""" # TODO: check unary and font-family STRING a, b, "c" val, type_ = self._value # try get by type_ pt = self.__unitbytype.get(type_, 'CSS_UNKNOWN') if callable(pt): # multiple options, check value too pt = pt(val) self._primitiveType = getattr(self, pt) def _getPrimitiveType(self): if not hasattr(self, '_primitivetype'): self.__set_primitiveType() return self._primitiveType primitiveType = property(_getPrimitiveType, doc="(readonly) The type of the value as defined " "by the constants in this class.") def _getPrimitiveTypeString(self): return self._unitnames[self.primitiveType] primitiveTypeString = property(_getPrimitiveTypeString, doc="Name of primitive type of this value.") def _getCSSPrimitiveTypeString(self, type): "get TypeString by given type which may be unknown, used by setters" try: return self._unitnames[type] except (IndexError, TypeError): return u'%r (UNKNOWN TYPE)' % type def _getNumDim(self, value=None): "Split self._value in numerical and dimension part." if value is None: value = cssutils.helper.normalize(self._value[0]) try: val, dim = CSSPrimitiveValue._reNumDim.findall(value)[0] except IndexError: val, dim = value, u'' try: val = float(val) if val == int(val): val = int(val) except ValueError: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: No float value %r' % self._value[0]) return val, dim def getFloatValue(self, unitType=None): """(DOM) This method is used to get a float value in a specified unit. If this CSS value doesn't contain a float value or can't be converted into the specified unit, a DOMException is raised. :param unitType: to get the float value. The unit code can only be a float unit type (i.e. CSS_NUMBER, CSS_PERCENTAGE, CSS_EMS, CSS_EXS, CSS_PX, CSS_CM, CSS_MM, CSS_IN, CSS_PT, CSS_PC, CSS_DEG, CSS_RAD, CSS_GRAD, CSS_MS, CSS_S, CSS_HZ, CSS_KHZ, CSS_DIMENSION) or None in which case the current dimension is used. :returns: not necessarily a float but some cases just an integer e.g. if the value is ``1px`` it return ``1`` and **not** ``1.0`` Conversions might return strange values like 1.000000000001 """ if unitType is not None and unitType not in self._floattypes: raise xml.dom.InvalidAccessErr( u'unitType Parameter is not a float type') val, dim = self._getNumDim() if unitType is not None and self.primitiveType != unitType: # convert if needed try: val = self._converter[self.primitiveType, unitType](val) except KeyError: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: Cannot coerce primitiveType %r to %r' % (self.primitiveTypeString, self._getCSSPrimitiveTypeString(unitType))) if val == int(val): val = int(val) return val def setFloatValue(self, unitType, floatValue): """(DOM) A method to set the float value with a specified unit. If the property attached with this value can not accept the specified unit or the float value, the value will be unchanged and a DOMException will be raised. :param unitType: a unit code as defined above. The unit code can only be a float unit type :param floatValue: the new float value which does not have to be a float value but may simple be an int e.g. if setting:: setFloatValue(CSS_PX, 1) :exceptions: - :exc:`~xml.dom.InvalidAccessErr`: Raised if the attached property doesn't support the float value or the unit type. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this property is readonly. """ self._checkReadonly() if unitType not in self._floattypes: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: unitType %r is not a float type' % self._getCSSPrimitiveTypeString(unitType)) try: val = float(floatValue) except ValueError, e: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: floatValue %r is not a float' % floatValue) oldval, dim = self._getNumDim() if self.primitiveType != unitType: # convert if possible try: val = self._converter[unitType, self.primitiveType](val) except KeyError: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: Cannot coerce primitiveType %r to %r' % (self.primitiveTypeString, self._getCSSPrimitiveTypeString(unitType))) if val == int(val): val = int(val) self.cssText = '%s%s' % (val, dim) def getStringValue(self): """(DOM) This method is used to get the string value. If the CSS value doesn't contain a string value, a DOMException is raised. Some properties (like 'font-family' or 'voice-family') convert a whitespace separated list of idents to a string. Only the actual value is returned so e.g. all the following return the actual value ``a``: url(a), attr(a), "a", 'a' """ if self.primitiveType not in self._stringtypes: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue %r is not a string type' % self.primitiveTypeString) if CSSPrimitiveValue.CSS_ATTR == self.primitiveType: return self._value[0].cssText[5:-1] else: return self._value[0] def setStringValue(self, stringType, stringValue): """(DOM) A method to set the string value with the specified unit. If the property attached to this value can't accept the specified unit or the string value, the value will be unchanged and a DOMException will be raised. :param stringType: a string code as defined above. The string code can only be a string unit type (i.e. CSS_STRING, CSS_URI, CSS_IDENT, and CSS_ATTR). :param stringValue: the new string value Only the actual value is expected so for (CSS_URI, "a") the new value will be ``url(a)``. For (CSS_STRING, "'a'") the new value will be ``"\\'a\\'"`` as the surrounding ``'`` are not part of the string value :exceptions: - :exc:`~xml.dom.InvalidAccessErr`: Raised if the CSS value doesn't contain a string value or if the string value can't be converted into the specified unit. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this property is readonly. """ self._checkReadonly() # self not stringType if self.primitiveType not in self._stringtypes: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue %r is not a string type' % self.primitiveTypeString) # given stringType is no StringType if stringType not in self._stringtypes: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: stringType %s is not a string type' % self._getCSSPrimitiveTypeString(stringType)) if self._primitiveType != stringType: raise xml.dom.InvalidAccessErr( u'CSSPrimitiveValue: Cannot coerce primitiveType %r to %r' % (self.primitiveTypeString, self._getCSSPrimitiveTypeString(stringType))) if CSSPrimitiveValue.CSS_STRING == self._primitiveType: self.cssText = cssutils.helper.string(stringValue) elif CSSPrimitiveValue.CSS_URI == self._primitiveType: self.cssText = cssutils.helper.uri(stringValue) elif CSSPrimitiveValue.CSS_ATTR == self._primitiveType: self.cssText = u'attr(%s)' % stringValue else: self.cssText = stringValue self._primitiveType = stringType def getCounterValue(self): """(DOM) This method is used to get the Counter value. If this CSS value doesn't contain a counter value, a DOMException is raised. Modification to the corresponding style property can be achieved using the Counter interface. **Not implemented.** """ if not self.CSS_COUNTER == self.primitiveType: raise xml.dom.InvalidAccessErr(u'Value is not a counter type') # TODO: use Counter class raise NotImplementedError() def getRGBColorValue(self): """(DOM) This method is used to get the RGB color. If this CSS value doesn't contain a RGB color value, a DOMException is raised. Modification to the corresponding style property can be achieved using the RGBColor interface. """ if self.primitiveType not in self._rbgtypes: raise xml.dom.InvalidAccessErr(u'Value is not a RGBColor value') return RGBColor(self._value[0]) def getRectValue(self): """(DOM) This method is used to get the Rect value. If this CSS value doesn't contain a rect value, a DOMException is raised. Modification to the corresponding style property can be achieved using the Rect interface. **Not implemented.** """ if self.primitiveType not in self._recttypes: raise xml.dom.InvalidAccessErr(u'value is not a Rect value') # TODO: use Rect class raise NotImplementedError() def _getCssText(self): """Overwrites CSSValue.""" return cssutils.ser.do_css_CSSPrimitiveValue(self) def _setCssText(self, cssText): """Use CSSValue.""" return super(CSSPrimitiveValue, self)._setCssText(cssText) cssText = property(_getCssText, _setCssText, doc="A string representation of the current value.") class CSSValueList(CSSValue): """The CSSValueList interface provides the abstraction of an ordered collection of CSS values. Some properties allow an empty list into their syntax. In that case, these properties take the none identifier. So, an empty list means that the property has the value none. The items in the CSSValueList are accessible via an integral index, starting from 0. """ cssValueType = CSSValue.CSS_VALUE_LIST def __init__(self, cssText=None, parent=None, readonly=False): """Init a new CSSValueList""" super(CSSValueList, self).__init__(cssText=cssText, parent=parent, readonly=readonly) self._items = [] def __iter__(self): "CSSValueList is iterable." for item in self.__items(): yield item.value def __str__(self): return u"" % (self.__class__.__name__, self.cssValueTypeString, self.cssText, self.length, id(self)) def __items(self): return [item for item in self._seq if isinstance(item.value, CSSValue)] def item(self, index): """(DOM) Retrieve a CSSValue by ordinal `index`. The order in this collection represents the order of the values in the CSS style property. If `index` is greater than or equal to the number of values in the list, this returns ``None``. """ try: return self.__items()[index].value except IndexError: return None length = property(lambda self: len(self.__items()), doc=u"(DOM attribute) The number of CSSValues in the " u"list.") class CSSFunction(CSSPrimitiveValue): """A CSS function value like rect() etc.""" _functionName = u'CSSFunction' primitiveType = CSSPrimitiveValue.CSS_UNKNOWN def __init__(self, cssText=None, parent=None, readonly=False): """ Init a new CSSFunction :param cssText: the parsable cssText of the value :param readonly: defaults to False """ super(CSSFunction, self).__init__(parent=parent) self._funcType = None self.valid = False self.wellformed = False if cssText is not None: self.cssText = cssText self._readonly = readonly def _productiondefinition(self): """Return definition used for parsing.""" types = self._prods # rename! value = Sequence(PreDef.unary(), Prod(name='PrimitiveValue', match=lambda t, v: t in (types.DIMENSION, types.HASH, types.IDENT, types.NUMBER, types.PERCENTAGE, types.STRING), toSeq=lambda t, tokens: (t[0], CSSPrimitiveValue(t[1])) ) ) valueOrFunc = Choice(value, # FUNC is actually not in spec but used in e.g. Prince PreDef.function(toSeq=lambda t, tokens: ('FUNCTION', CSSFunction( cssutils.helper.pushtoken(t, tokens)) ) ) ) funcProds = Sequence(Prod(name='FUNC', match=lambda t, v: t == types.FUNCTION, toSeq=lambda t, tokens: (t[0], cssutils.helper.normalize(t[1]))), Choice(Sequence(valueOrFunc, # more values starting with Comma # should use store where colorType is saved to # define min and may, closure? Sequence(PreDef.comma(), valueOrFunc, minmax=lambda: (0, None)), PreDef.funcEnd(stop=True)), PreDef.funcEnd(stop=True)) ) return funcProds def _setCssText(self, cssText): self._checkReadonly() # store: colorType, parts wellformed, seq, store, unusedtokens = ProdParser().parse(cssText, self._functionName, self._productiondefinition(), keepS=True) if wellformed: # combine +/- and following CSSPrimitiveValue, remove S newseq = self._tempSeq() i, end = 0, len(seq) while i < end: item = seq[i] if item.type == self._prods.S: pass elif item.value == u'+' or item.value == u'-': i += 1 next = seq[i] newval = next.value if isinstance(newval, CSSPrimitiveValue): newval.setFloatValue(newval.primitiveType, float(item.value + str(newval.getFloatValue()))) newseq.append(newval, next.type, item.line, item.col) else: # expressions only? newseq.appendItem(item) newseq.appendItem(next) else: newseq.appendItem(item) i += 1 self.wellformed = True self._setSeq(newseq) self._funcType = newseq[0].value cssText = property(lambda self: cssutils.ser.do_css_FunctionValue(self), _setCssText) funcType = property(lambda self: self._funcType) class RGBColor(CSSFunction): """A CSS color like RGB, RGBA or a simple value like `#000` or `red`.""" _functionName = u'Function rgb()' def __init__(self, cssText=None, parent=None, readonly=False): """ Init a new RGBColor :param cssText: the parsable cssText of the value :param readonly: defaults to False """ super(CSSFunction, self).__init__(parent=parent) self._colorType = None self.valid = False self.wellformed = False if cssText is not None: try: # if it is a Function object cssText = cssText.cssText except AttributeError: pass self.cssText = cssText self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(%r)" % (self.__class__.__name__, self.cssText) def __str__(self): return u"" % ( self.__class__.__name__, self.colorType, self.cssText, id(self)) def _setCssText(self, cssText): self._checkReadonly() types = self._prods # rename! valueProd = Prod(name='value', match=lambda t, v: t in (types.NUMBER, types.PERCENTAGE), toSeq=lambda t, v: (CSSPrimitiveValue, CSSPrimitiveValue(v)), toStore='parts' ) # COLOR PRODUCTION funccolor = Sequence(Prod(name='FUNC', match=lambda t, v: t == types.FUNCTION and cssutils.helper.normalize(v) in ('rgb(', 'rgba(', 'hsl(', 'hsla('), toSeq=lambda t, v: (t, v),#cssutils.helper.normalize(v)), toStore='colorType'), PreDef.unary(), valueProd, # 2 or 3 more values starting with Comma Sequence(PreDef.comma(), PreDef.unary(), valueProd, minmax=lambda: (2, 3)), PreDef.funcEnd() ) colorprods = Choice(funccolor, PreDef.hexcolor('colorType'), Prod(name='named color', match=lambda t, v: t == types.IDENT, toStore='colorType' ) ) # store: colorType, parts wellformed, seq, store, unusedtokens = ProdParser().parse(cssText, u'RGBColor', colorprods, keepS=True, store={'parts': []}) if wellformed: self.wellformed = True if store['colorType'].type == self._prods.HASH: self._colorType = 'HEX' elif store['colorType'].type == self._prods.IDENT: self._colorType = 'Named Color' else: self._colorType = store['colorType'].value[:-1] #self._colorType = cssutils.helper.normalize(store['colorType'].value)[:-1] self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_RGBColor(self), _setCssText) colorType = property(lambda self: self._colorType) class CalcValue(CSSFunction): """Calc Function""" _functionName = u'Function calc()' def _productiondefinition(self): """Return defintion used for parsing.""" types = self._prods # rename! def toSeq(t, tokens): "Do not normalize function name!" return t[0], t[1] funcProds = Sequence(Prod(name='calc', match=lambda t, v: t == types.FUNCTION, toSeq=toSeq ), Sequence(Choice(Prod(name='nested function', match=lambda t, v: t == self._prods.FUNCTION, toSeq=lambda t, tokens: (CSSFunction._functionName, CSSFunction(cssutils.helper.pushtoken(t, tokens))) ), Prod(name='part', match=lambda t, v: v != u')', toSeq=lambda t, tokens: (t[0], t[1])), ), minmax=lambda: (0, None)), PreDef.funcEnd(stop=True)) return funcProds def _getCssText(self): return cssutils.ser.do_css_CalcValue(self) def _setCssText(self, cssText): return super(CalcValue, self)._setCssText(cssText) cssText = property(_getCssText, _setCssText, doc=u"A string representation of the current value.") class ExpressionValue(CSSFunction): """Special IE only CSSFunction which may contain *anything*. Used for expressions and ``alpha(opacity=100)`` currently.""" _functionName = u'Expression (IE only)' def _productiondefinition(self): """Return defintion used for parsing.""" types = self._prods # rename! def toSeq(t, tokens): "Do not normalize function name!" return t[0], t[1] funcProds = Sequence(Prod(name='expression', match=lambda t, v: t == types.FUNCTION, toSeq=toSeq ), Sequence(Choice(Prod(name='nested function', match=lambda t, v: t == self._prods.FUNCTION, toSeq=lambda t, tokens: (ExpressionValue._functionName, ExpressionValue(cssutils.helper.pushtoken(t, tokens))) ), Prod(name='part', match=lambda t, v: v != u')', toSeq=lambda t, tokens: (t[0], t[1])), ), minmax=lambda: (0, None)), PreDef.funcEnd(stop=True)) return funcProds def _getCssText(self): return cssutils.ser.do_css_ExpressionValue(self) def _setCssText(self, cssText): #self._log.warn(u'CSSValue: Unoffial and probably invalid MS value used!') return super(ExpressionValue, self)._setCssText(cssText) cssText = property(_getCssText, _setCssText, doc=u"A string representation of the current value.") class CSSVariable(CSSValue): """The CSSVariable represents a call to CSS Variable.""" def __init__(self, cssText=None, parent=None, readonly=False): """Init a new CSSVariable. :param cssText: the parsable cssText of the value, e.g. ``var(x)`` :param readonly: defaults to False """ self._name = None super(CSSVariable, self).__init__(cssText=cssText, parent=parent, readonly=readonly) def __repr__(self): return u"cssutils.css.%s(%r)" % (self.__class__.__name__, self.cssText) def __str__(self): return u"" % ( self.__class__.__name__, self.name, self.value, id(self)) def _setCssText(self, cssText): self._checkReadonly() types = self._prods # rename! funcProds = Sequence(Prod(name='var', match=lambda t, v: t == types.FUNCTION ), PreDef.ident(toStore='ident'), PreDef.funcEnd(stop=True)) # store: name of variable store = {'ident': None} wellformed, seq, store, unusedtokens = ProdParser().parse(cssText, u'CSSVariable', funcProds, keepS=True) if wellformed: self._name = store['ident'].value self._setSeq(seq) self.wellformed = True cssText = property(lambda self: cssutils.ser.do_css_CSSVariable(self), _setCssText, doc=u"A string representation of the current variable.") cssValueType = CSSValue.CSS_VARIABLE # TODO: writable? check if var (value) available? name = property(lambda self: self._name) def _getValue(self): "Find contained sheet and @variables there" try: variables = self.parent.parent.parentRule.parentStyleSheet.variables except AttributeError: return None else: try: return variables[self.name] except KeyError: return None value = property(_getValue) cssutils-1.0/src/cssutils/css/value.py0000666000175000017500000011374512253154606016652 0ustar hugohugo"""Value related classes. DOM Level 2 CSS CSSValue, CSSPrimitiveValue and CSSValueList are **no longer** supported and are replaced by these new classes. """ __all__ = ['PropertyValue', 'Value', 'ColorValue', 'DimensionValue', 'URIValue', 'CSSFunction', 'CSSCalc', 'CSSVariable', 'MSValue' ] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.prodparser import * import cssutils from cssutils.helper import normalize, pushtoken import colorsys import math import re import xml.dom import urlparse class PropertyValue(cssutils.util._NewBase): """ An unstructured list like holder for all values defined for a :class:`~cssutils.css.Property`. Contains :class:`~cssutils.css.Value` or subclass objects. Currently there is no access to the combinators of the defined values which might simply be space or comma or slash. You may: - iterate over all contained Value objects (not the separators like ``,``, ``/`` or `` `` though!) - get a Value item by index or use ``PropertyValue[index]`` - find out the number of values defined (unstructured) """ def __init__(self, cssText=None, parent=None, readonly=False): """ :param cssText: the parsable cssText of the value :param readonly: defaults to False """ super(PropertyValue, self).__init__() self.parent = parent self.wellformed = False if cssText is not None: # may be 0 if isinstance(cssText, (int, float)): cssText = unicode(cssText) # if it is a number self.cssText = cssText self._readonly = readonly def __len__(self): return len(list(self.__items())) def __getitem__(self, index): try: return list(self.__items())[index] except IndexError: return None def __iter__(self): "Generator which iterates over values." for item in self.__items(): yield item def __repr__(self): return u"cssutils.css.%s(%r)" % (self.__class__.__name__, self.cssText) def __str__(self): return u"" % (self.__class__.__name__, self.length, self.cssText, id(self)) def __items(self, seq=None): "a generator of Value obects only, no , / or ' '" if seq is None: seq = self.seq return (x.value for x in seq if isinstance(x.value, Value)) def _setCssText(self, cssText): if isinstance(cssText, (int, float)): cssText = unicode(cssText) # if it is a number """ Format:: unary_operator : '-' | '+' ; operator : '/' S* | ',' S* | /* empty */ ; expr : term [ operator term ]* ; term : unary_operator? [ NUMBER S* | PERCENTAGE S* | LENGTH S* | EMS S* | EXS S* | ANGLE S* | TIME S* | FREQ S* ] | STRING S* | IDENT S* | URI S* | hexcolor | function | UNICODE-RANGE S* ; function : FUNCTION S* expr ')' S* ; /* * There is a constraint on the color that it must * have either 3 or 6 hex-digits (i.e., [0-9a-fA-F]) * after the "#"; e.g., "#000" is OK, but "#abcd" is not. */ hexcolor : HASH S* ; :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error (according to the attached property) or is unparsable. - :exc:`~xml.dom.InvalidModificationErr`: TODO: Raised if the specified CSS string value represents a different type of values than the values allowed by the CSS property. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this value is readonly. """ self._checkReadonly() # used as operator is , / or S nextSor = u',/' term = Choice(_ColorProd(self, nextSor), _DimensionProd(self, nextSor), _URIProd(self, nextSor), _ValueProd(self, nextSor), # _Rect(self, nextSor), # all other functions _CSSVariableProd(self, nextSor), _MSValueProd(self, nextSor), _CalcValueProd(self, nextSor), _CSSFunctionProd(self, nextSor) ) operator = Choice(PreDef.S(toSeq=False), PreDef.char('comma', ',', toSeq=lambda t, tokens: ('operator', t[1]), optional=True ), PreDef.char('slash', '/', toSeq=lambda t, tokens: ('operator', t[1]), optional=True), optional=True) prods = Sequence(term, Sequence(# mayEnd this Sequence if whitespace operator, # TODO: only when setting via other class # used by variabledeclaration currently PreDef.char('END', ';', stopAndKeep=True, optional=True), # TODO: } and !important ends too! term, minmax=lambda: (0, None))) # parse ok, seq, store, unused = ProdParser().parse(cssText, u'PropertyValue', prods) # must be at least one value! ok = ok and len(list(self.__items(seq))) > 0 for item in seq: if hasattr(item.value, 'wellformed') and not item.value.wellformed: ok = False break self.wellformed = ok if ok: self._setSeq(seq) else: self._log.error(u'PropertyValue: Unknown syntax or no value: %s' % self._valuestr(cssText)) cssText = property(lambda self: cssutils.ser.do_css_PropertyValue(self), _setCssText, doc="A string representation of the current value.") def item(self, index): """ The value at position `index`. Alternatively simple use ``PropertyValue[index]``. :param index: the parsable cssText of the value :exceptions: - :exc:`~IndexError`: Raised if index if out of bounds """ return self[index] length = property(lambda self: len(self), doc=u"Number of values set.") value = property(lambda self: cssutils.ser.do_css_PropertyValue(self, valuesOnly=True), doc=u"A string representation of the current value " u"without any comments used for validation.") class Value(cssutils.util._NewBase): """ Represents a single CSS value. For now simple values of IDENT, STRING, or UNICODE-RANGE values are represented directly as Value objects. Other values like e.g. FUNCTIONs are represented by subclasses with an extended API. """ IDENT = u'IDENT' STRING = u'STRING' UNICODE_RANGE = u'UNICODE-RANGE' URI = u'URI' DIMENSION = u'DIMENSION' NUMBER = u'NUMBER' PERCENTAGE = u'PERCENTAGE' COLOR_VALUE = u'COLOR_VALUE' HASH = u'HASH' FUNCTION = u'FUNCTION' CALC = u'CALC' VARIABLE = u'VARIABLE' _type = None _value = u'' def __init__(self, cssText=None, parent=None, readonly=False): super(Value, self).__init__() self.parent = parent self.wellformed = False if cssText: self.cssText = cssText def __repr__(self): return u"cssutils.css.%s(%r)" % (self.__class__.__name__, self.cssText) def __str__(self): return u""\ % (self.__class__.__name__, self.type, self.value, self.cssText, id(self)) def _setCssText(self, cssText): self._checkReadonly() prods = Choice(PreDef.hexcolor(stop=True), PreDef.ident(stop=True), PreDef.string(stop=True), PreDef.unicode_range(stop=True), ) ok, seq, store, unused = ProdParser().parse(cssText, u'Value', prods) self.wellformed = ok if ok: # only 1 value anyway! self._type = seq[0].type self._value = seq[0].value self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_Value(self), _setCssText, doc=u'String value of this value.') type = property(lambda self: self._type, #_setType, doc=u"Type of this value, for now the production type " u"like e.g. `DIMENSION` or `STRING`. All types are " u"defined as constants in :class:`~cssutils.css.Value`.") def _setValue(self, value): # TODO: check! self._value = value value = property(lambda self: self._value, _setValue, doc=u"Actual value if possible: An int or float or else " u" a string") class ColorValue(Value): """ A color value like rgb(), rgba(), hsl(), hsla() or #rgb, #rrggbb TODO: Color Keywords """ from colors import COLORS type = Value.COLOR_VALUE # hexcolor, FUNCTION? _colorType = None _red = 0 _green = 0 _blue = 0 _alpha = 0 def __str__(self): return u""\ % (self.__class__.__name__, self.type, self.value, self.colorType, self.red, self.green, self.blue, self.alpha, id(self)) def _setCssText(self, cssText): self._checkReadonly() types = self._prods # rename! component = Choice(PreDef.unary(toSeq=lambda t, tokens: (t[0], DimensionValue(pushtoken(t, tokens), parent=self) )), PreDef.number(toSeq=lambda t, tokens: (t[0], DimensionValue(pushtoken(t, tokens), parent=self) )), PreDef.percentage(toSeq=lambda t, tokens: (t[0], DimensionValue(pushtoken(t, tokens), parent=self) )) ) noalp = Sequence(Prod(name='FUNCTION', match=lambda t, v: t == types.FUNCTION and v in (u'rgb(', u'hsl('), toSeq=lambda t, tokens: (t[0], normalize(t[1]))), component, Sequence(PreDef.comma(optional=True), component, minmax=lambda: (2, 2) ), PreDef.funcEnd(stop=True) ) witha = Sequence(Prod(name='FUNCTION', match=lambda t, v: t == types.FUNCTION and v in (u'rgba(', u'hsla('), toSeq=lambda t, tokens: (t[0], normalize(t[1])) ), component, Sequence(PreDef.comma(optional=True), component, minmax=lambda: (3, 3) ), PreDef.funcEnd(stop=True) ) namedcolor = Prod(name='Named Color', match=lambda t, v: t == 'IDENT' and ( normalize(v) in self.COLORS.keys() ), stop=True) prods = Choice(PreDef.hexcolor(stop=True), namedcolor, noalp, witha) ok, seq, store, unused = ProdParser().parse(cssText, self.type, prods) self.wellformed = ok if ok: t, v = seq[0].type, seq[0].value if u'IDENT' == t: rgba = self.COLORS[normalize(v)] if u'HASH' == t: if len(v) == 4: # HASH #rgb rgba = (int(2*v[1], 16), int(2*v[2], 16), int(2*v[3], 16), 1.0) else: # HASH #rrggbb rgba = (int(v[1:3], 16), int(v[3:5], 16), int(v[5:7], 16), 1.0) elif u'FUNCTION' == t: functiontype, raw, check = None, [], u'' HSL = False for item in seq: try: type_ = item.value.type except AttributeError, e: # type of function, e.g. rgb( if item.type == 'FUNCTION': functiontype = item.value HSL = functiontype in (u'hsl(', u'hsla(') continue # save components if type_ == Value.NUMBER: raw.append(item.value.value) check += u'N' elif type_ == Value.PERCENTAGE: if HSL: # save as percentage fraction raw.append(item.value.value / 100.0) else: # save as real value of percentage of 255 raw.append(int(255 * item.value.value / 100)) check += u'P' if HSL: # convert to rgb # h is 360 based (circle) h, s, l = raw[0] / 360.0, raw[1], raw[2] # ORDER h l s !!! r, g, b = colorsys.hls_to_rgb(h, l, s) # back to 255 based rgba = [int(round(r*255)), int(round(g*255)), int(round(b*255))] if len(raw) > 3: rgba.append(raw[3]) else: # rgb, rgba rgba = raw if len(rgba) < 4: rgba.append(1.0) # validate checks = {u'rgb(': ('NNN', 'PPP'), u'rgba(': ('NNNN', 'PPPN'), u'hsl(': ('NPP',), u'hsla(': ('NPPN',) } if check not in checks[functiontype]: self._log.error(u'ColorValue has invalid %s) parameters: ' u'%s (N=Number, P=Percentage)' % (functiontype, check)) self._colorType = t self._red, self._green, self._blue, self._alpha = tuple(rgba) self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_ColorValue(self), _setCssText, doc=u"String value of this value.") value = property(lambda self: cssutils.ser.do_css_CSSFunction(self, True), doc=u'Same as cssText but without comments.') type = property(lambda self: Value.COLOR_VALUE, doc=u"Type is fixed to Value.COLOR_VALUE.") def _getName(self): for n, v in self.COLORS.items(): if v == (self.red, self.green, self.blue, self.alpha): return n colorType = property(lambda self: self._colorType, doc=u"IDENT (red), HASH (#f00) or FUNCTION (rgb(255, 0, 0).") name = property(_getName, doc=u'Name of the color if known (in ColorValue.COLORS) ' u'else None') red = property(lambda self: self._red, doc=u'red part as integer between 0 and 255') green = property(lambda self: self._green, doc=u'green part as integer between 0 and 255') blue = property(lambda self: self._blue, doc=u'blue part as integer between 0 and 255') alpha = property(lambda self: self._alpha, doc=u'alpha part as float between 0.0 and 1.0') class DimensionValue(Value): """ A numerical value with an optional dimension like e.g. "px" or "%". Covers DIMENSION, PERCENTAGE or NUMBER values. """ __reUnNumDim = re.compile(ur'^([+-]?)(\d*\.\d+|\d+)(.*)$', re.I | re.U | re.X) _dimension = None _sign = None def __str__(self): return u""\ % (self.__class__.__name__, self.type, self.value, self.dimension, self.cssText, id(self)) def _setCssText(self, cssText): self._checkReadonly() prods = Sequence(#PreDef.unary(), Choice(PreDef.dimension(stop=True), PreDef.number(stop=True), PreDef.percentage(stop=True) ) ) ok, seq, store, unused = ProdParser().parse(cssText, u'DimensionValue', prods) self.wellformed = ok if ok: item = seq[0] sign, v, d = self.__reUnNumDim.findall( normalize(item.value))[0] if u'.' in v: val = float(sign + v) else: val = int(sign + v) dim = None if d: dim = d self._sign = sign self._value = val self._dimension = dim self._type = item.type self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_Value(self), _setCssText, doc=u"String value of this value including dimension.") dimension = property(lambda self: self._dimension, #_setValue, doc=u"Dimension if a DIMENSION or PERCENTAGE value, " u"else None") class URIValue(Value): """ An URI value like ``url(example.png)``. """ _type = Value.URI _uri = Value._value def __str__(self): return u""\ % (self.__class__.__name__, self.type, self.value, self.uri, self.cssText, id(self)) def _setCssText(self, cssText): self._checkReadonly() prods = Sequence(PreDef.uri(stop=True)) ok, seq, store, unused = ProdParser().parse(cssText, u'URIValue', prods) self.wellformed = ok if ok: # only 1 value only anyway self._type = seq[0].type self._value = seq[0].value self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_Value(self), _setCssText, doc=u'String value of this value.') def _setUri(self, uri): # TODO: check? self._value = uri uri = property(lambda self: self._value, _setUri, doc=u"Actual URL without delimiters or the empty string") def absoluteUri(self): """Actual URL, made absolute if possible, else same as `uri`.""" # Ancestry: PropertyValue, Property, CSSStyleDeclaration, CSSStyleRule, # CSSStyleSheet try: # TODO: better way? styleSheet = self.parent.parent.parent.parentRule.parentStyleSheet except AttributeError, e: return self.uri else: return urlparse.urljoin(styleSheet.href, self.uri) absoluteUri = property(absoluteUri, doc=absoluteUri.__doc__) class CSSFunction(Value): """ A function value. """ _functionName = 'Function' def _productions(self): """Return definition used for parsing.""" types = self._prods # rename! itemProd = Choice(_ColorProd(self), _DimensionProd(self), _URIProd(self), _ValueProd(self), _CalcValueProd(self), _CSSVariableProd(self), _CSSFunctionProd(self) ) funcProds = Sequence(Prod(name='FUNCTION', match=lambda t, v: t == types.FUNCTION, toSeq=lambda t, tokens: (t[0], normalize(t[1]))), Choice(Sequence(itemProd, Sequence(PreDef.comma(optional=True), itemProd, minmax=lambda: (0, None)), PreDef.funcEnd(stop=True)), PreDef.funcEnd(stop=True)) ) return funcProds def _setCssText(self, cssText): self._checkReadonly() ok, seq, store, unused = ProdParser().parse(cssText, self.type, self._productions()) self.wellformed = ok if ok: self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_CSSFunction(self), _setCssText, doc=u"String value of this value.") value = property(lambda self: cssutils.ser.do_css_CSSFunction(self, True), doc=u'Same as cssText but without comments.') type = property(lambda self: Value.FUNCTION, doc=u"Type is fixed to Value.FUNCTION.") class MSValue(CSSFunction): """An IE specific Microsoft only function value which is much looser in what is syntactically allowed.""" _functionName = 'MSValue' def _productions(self): """Return definition used for parsing.""" types = self._prods # rename! func = Prod(name='MSValue-Sub', match=lambda t, v: t == self._prods.FUNCTION, toSeq=lambda t, tokens: (MSValue._functionName, MSValue(pushtoken(t, tokens ), parent=self ) ) ) funcProds = Sequence(Prod(name='FUNCTION', match=lambda t, v: t == types.FUNCTION, toSeq=lambda t, tokens: (t[0], t[1]) ), Sequence(Choice(_ColorProd(self), _DimensionProd(self), _URIProd(self), _ValueProd(self), _MSValueProd(self), #_CalcValueProd(self), _CSSVariableProd(self), func, #_CSSFunctionProd(self), Prod(name='MSValuePart', match=lambda t, v: v != u')', toSeq=lambda t, tokens: (t[0], t[1]) ) ), minmax=lambda: (0, None) ), PreDef.funcEnd(stop=True) ) return funcProds def _setCssText(self, cssText): super(MSValue, self)._setCssText(cssText) cssText = property(lambda self: cssutils.ser.do_css_MSValue(self), _setCssText, doc=u"String value of this value.") class CSSCalc(CSSFunction): """The CSSCalc function represents a CSS calc() function. No further API is provided. For multiplication and division no check if one operand is a NUMBER is made. """ _functionName = 'CSSCalc' def __str__(self): return u"" % ( self.__class__.__name__, id(self)) def _setCssText(self, cssText): self._checkReadonly() types = self._prods # rename! _operator = Choice(Prod(name='Operator */', match=lambda t, v: v in u'*/', toSeq=lambda t, tokens: (t[0], t[1]) ), Sequence( PreDef.S(), Choice( Sequence( Prod(name='Operator */', match=lambda t, v: v in u'*/', toSeq=lambda t, tokens: (t[0], t[1]) ), PreDef.S(optional=True) ), Sequence( Prod(name='Operator +-', match=lambda t, v: v in u'+-', toSeq=lambda t, tokens: (t[0], t[1]) ), PreDef.S() ), PreDef.funcEnd(stop=True, mayEnd=True) ) ) ) _operant = lambda: Choice(_DimensionProd(self), _CSSVariableProd(self)) prods = Sequence(Prod(name='CALC', match=lambda t, v: t == types.FUNCTION and normalize(v) == u'calc(' ), PreDef.S(optional=True), _operant(), Sequence(_operator, _operant(), minmax=lambda: (0, None) ), PreDef.funcEnd(stop=True) ) # store: name of variable ok, seq, store, unused = ProdParser().parse(cssText, u'CSSCalc', prods, checkS=True) self.wellformed = ok if ok: self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_CSSCalc(self), _setCssText, doc=u"String representation of calc function.") type = property(lambda self: Value.CALC, doc=u"Type is fixed to Value.CALC.") class CSSVariable(CSSFunction): """The CSSVariable represents a CSS variables like ``var(varname)``. A variable has a (nonnormalized!) `name` and a `value` which is tried to be resolved from any available CSSVariablesRule definition. """ _functionName = 'CSSVariable' _name = None _fallback = None def __str__(self): return u"" % ( self.__class__.__name__, self.name, self.value, id(self)) def _setCssText(self, cssText): self._checkReadonly() types = self._prods # rename! prods = Sequence(Prod(name='var', match=lambda t, v: t == types.FUNCTION and normalize(v) == u'var(' ), PreDef.ident(toStore='ident'), Sequence(PreDef.comma(), Choice(_ColorProd(self, toStore='fallback'), _DimensionProd(self, toStore='fallback'), _URIProd(self, toStore='fallback'), _ValueProd(self, toStore='fallback'), _CalcValueProd(self, toStore='fallback'), _CSSVariableProd(self, toStore='fallback'), _CSSFunctionProd(self, toStore='fallback') ), minmax=lambda: (0, 1) ), PreDef.funcEnd(stop=True)) # store: name of variable store = {'ident': None, 'fallback': None} ok, seq, store, unused = ProdParser().parse(cssText, u'CSSVariable', prods) self.wellformed = ok if ok: self._name = store['ident'].value try: self._fallback = store['fallback'].value except KeyError: self._fallback = None self._setSeq(seq) cssText = property(lambda self: cssutils.ser.do_css_CSSVariable(self), _setCssText, doc=u"String representation of variable.") # TODO: writable? check if var (value) available? name = property(lambda self: self._name, doc=u"The name identifier of this variable referring to " u"a value in a " u":class:`cssutils.css.CSSVariablesDeclaration`.") fallback = property(lambda self: self._fallback, doc=u"The fallback Value of this variable") type = property(lambda self: Value.VARIABLE, doc=u"Type is fixed to Value.VARIABLE.") def _getValue(self): "Find contained sheet and @variables there" rel = self while True: # find node which has parentRule to get to StyleSheet if hasattr(rel, 'parent'): rel = rel.parent else: break try: variables = rel.parentRule.parentStyleSheet.variables except AttributeError: return None else: try: return variables[self.name] except KeyError: return None value = property(_getValue, doc=u'The resolved actual value or None.') # helper for productions def _ValueProd(parent, nextSor=False, toStore=None): return Prod(name='Value', match=lambda t, v: t in ('IDENT', 'STRING', 'UNICODE-RANGE'), nextSor = nextSor, toStore=toStore, toSeq=lambda t, tokens: ('Value', Value(pushtoken(t, tokens), parent=parent) ) ) def _DimensionProd(parent, nextSor=False, toStore=None): return Prod(name='Dimension', match=lambda t, v: t in (u'DIMENSION', u'NUMBER', u'PERCENTAGE'), nextSor = nextSor, toStore=toStore, toSeq=lambda t, tokens: (u'DIMENSION', DimensionValue( pushtoken(t, tokens), parent=parent) ) ) def _URIProd(parent, nextSor=False, toStore=None): return Prod(name='URIValue', match=lambda t, v: t == 'URI', toStore=toStore, nextSor = nextSor, toSeq=lambda t, tokens: ('URIValue', URIValue( pushtoken(t, tokens), parent=parent) ) ) reHexcolor = re.compile(r'^\#(?:[0-9abcdefABCDEF]{3}|[0-9abcdefABCDEF]{6})$') def _ColorProd(parent, nextSor=False, toStore=None): return Prod(name='ColorValue', match=lambda t, v: (t == 'HASH' and reHexcolor.match(v) ) or (t == 'FUNCTION' and normalize(v) in (u'rgb(', u'rgba(', u'hsl(', u'hsla(') ) or (t == 'IDENT' and normalize(v) in ColorValue.COLORS.keys() ), nextSor = nextSor, toStore=toStore, toSeq=lambda t, tokens: ('ColorValue', ColorValue( pushtoken(t, tokens), parent=parent) ) ) def _CSSFunctionProd(parent, nextSor=False, toStore=None): return PreDef.function(nextSor=nextSor, toStore=toStore, toSeq=lambda t, tokens: (CSSFunction._functionName, CSSFunction( pushtoken(t, tokens), parent=parent) ) ) def _CalcValueProd(parent, nextSor=False, toStore=None): return Prod(name=CSSCalc._functionName, match=lambda t, v: t == PreDef.types.FUNCTION and normalize(v) == u'calc(', toStore=toStore, toSeq=lambda t, tokens: (CSSCalc._functionName, CSSCalc( pushtoken(t, tokens), parent=parent) ), nextSor=nextSor) def _CSSVariableProd(parent, nextSor=False, toStore=None): return PreDef.variable(nextSor=nextSor, toStore=toStore, toSeq=lambda t, tokens: (CSSVariable._functionName, CSSVariable( pushtoken(t, tokens), parent=parent) ) ) def _MSValueProd(parent, nextSor=False): return Prod(name=MSValue._functionName, match=lambda t, v: (#t == self._prods.FUNCTION and ( normalize(v) in (u'expression(', u'alpha(', u'blur(', u'chroma(', u'dropshadow(', u'fliph(', u'flipv(', u'glow(', u'gray(', u'invert(', u'mask(', u'shadow(', u'wave(', u'xray(') or v.startswith(u'progid:DXImageTransform.Microsoft.') ), nextSor=nextSor, toSeq=lambda t, tokens: (MSValue._functionName, MSValue(pushtoken(t, tokens ), parent=parent ) ) ) def MediaQueryValueProd(parent): return Choice(_ColorProd(parent), _DimensionProd(parent), _ValueProd(parent), )cssutils-1.0/src/cssutils/css/cssstylesheet.py0000666000175000017500000010052012126054542020420 0ustar hugohugo"""CSSStyleSheet implements DOM Level 2 CSS CSSStyleSheet. Partly also: - http://dev.w3.org/csswg/cssom/#the-cssstylesheet - http://www.w3.org/TR/2006/WD-css3-namespace-20060828/ TODO: - ownerRule and ownerNode """ __all__ = ['CSSStyleSheet'] __docformat__ = 'restructuredtext' __version__ = '$Id$' from cssutils.helper import Deprecated from cssutils.util import _Namespaces, _SimpleNamespaces, _readUrl from cssrule import CSSRule from cssvariablesdeclaration import CSSVariablesDeclaration import cssutils.stylesheets import xml.dom class CSSStyleSheet(cssutils.stylesheets.StyleSheet): """CSSStyleSheet represents a CSS style sheet. Format:: stylesheet : [ CHARSET_SYM S* STRING S* ';' ]? [S|CDO|CDC]* [ import [S|CDO|CDC]* ]* [ namespace [S|CDO|CDC]* ]* # according to @namespace WD [ [ ruleset | media | page ] [S|CDO|CDC]* ]* ``cssRules`` All Rules in this style sheet, a :class:`~cssutils.css.CSSRuleList`. """ def __init__(self, href=None, media=None, title=u'', disabled=None, ownerNode=None, parentStyleSheet=None, readonly=False, ownerRule=None, validating=True): """ For parameters see :class:`~cssutils.stylesheets.StyleSheet` """ super(CSSStyleSheet, self).__init__( 'text/css', href, media, title, disabled, ownerNode, parentStyleSheet, validating=validating) self._ownerRule = ownerRule self.cssRules = cssutils.css.CSSRuleList() self._namespaces = _Namespaces(parentStyleSheet=self, log=self._log) self._variables = CSSVariablesDeclaration() self._readonly = readonly # used only during setting cssText by parse*() self.__encodingOverride = None self._fetcher = None def __iter__(self): "Generator which iterates over cssRules." for rule in self._cssRules: yield rule def __repr__(self): if self.media: mediaText = self.media.mediaText else: mediaText = None return "cssutils.css.%s(href=%r, media=%r, title=%r)" % ( self.__class__.__name__, self.href, mediaText, self.title) def __str__(self): if self.media: mediaText = self.media.mediaText else: mediaText = None return "" % ( self.__class__.__name__, self.encoding, self.href, mediaText, self.title, self.namespaces.namespaces, id(self)) def _cleanNamespaces(self): "Remove all namespace rules with same namespaceURI but last." rules = self.cssRules namespaceitems = self.namespaces.items() i = 0 while i < len(rules): rule = rules[i] if rule.type == rule.NAMESPACE_RULE and \ (rule.prefix, rule.namespaceURI) not in namespaceitems: self.deleteRule(i) else: i += 1 def _getUsedURIs(self): "Return set of URIs used in the sheet." useduris = set() for r1 in self: if r1.STYLE_RULE == r1.type: useduris.update(r1.selectorList._getUsedUris()) elif r1.MEDIA_RULE == r1.type: for r2 in r1: if r2.type == r2.STYLE_RULE: useduris.update(r2.selectorList._getUsedUris()) return useduris def _setCssRules(self, cssRules): "Set new cssRules and update contained rules refs." cssRules.append = self.insertRule cssRules.extend = self.insertRule cssRules.__delitem__ = self.deleteRule for rule in cssRules: rule._parentStyleSheet = self self._cssRules = cssRules cssRules = property(lambda self: self._cssRules, _setCssRules, u"All Rules in this style sheet, a " u":class:`~cssutils.css.CSSRuleList`.") def _getCssText(self): "Textual representation of the stylesheet (a byte string)." return cssutils.ser.do_CSSStyleSheet(self) def _setCssText(self, cssText): """Parse `cssText` and overwrites the whole stylesheet. :param cssText: a parseable string or a tuple of (cssText, dict-of-namespaces) :exceptions: - :exc:`~xml.dom.NamespaceErr`: If a namespace prefix is found which is not declared. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. """ self._checkReadonly() cssText, namespaces = self._splitNamespacesOff(cssText) tokenizer = self._tokenize2(cssText) def S(expected, seq, token, tokenizer=None): # @charset must be at absolute beginning of style sheet # or 0 for py3 return max(1, expected or 0) def COMMENT(expected, seq, token, tokenizer=None): "special: sets parent*" self.insertRule(cssutils.css.CSSComment([token], parentStyleSheet=self)) # or 0 for py3 return max(1, expected or 0) def charsetrule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSCharsetRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if expected > 0: self._log.error(u'CSSStylesheet: CSSCharsetRule only allowed ' u'at beginning of stylesheet.', token, xml.dom.HierarchyRequestErr) return expected elif rule.wellformed: self.insertRule(rule) return 1 def importrule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSImportRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if expected > 1: self._log.error(u'CSSStylesheet: CSSImportRule not allowed ' u'here.', token, xml.dom.HierarchyRequestErr) return expected elif rule.wellformed: self.insertRule(rule) return 1 def namespacerule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSNamespaceRule(cssText=self._tokensupto2(tokenizer, token), parentStyleSheet=self) if expected > 2: self._log.error(u'CSSStylesheet: CSSNamespaceRule not allowed ' u'here.', token, xml.dom.HierarchyRequestErr) return expected elif rule.wellformed: if rule.prefix not in self.namespaces: # add new if not same prefix self.insertRule(rule, _clean=False) else: # same prefix => replace namespaceURI for r in self.cssRules.rulesOfType(rule.NAMESPACE_RULE): if r.prefix == rule.prefix: r._replaceNamespaceURI(rule.namespaceURI) self._namespaces[rule.prefix] = rule.namespaceURI return 2 def variablesrule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSVariablesRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if expected > 2: self._log.error(u'CSSStylesheet: CSSVariablesRule not allowed ' u'here.', token, xml.dom.HierarchyRequestErr) return expected elif rule.wellformed: self.insertRule(rule) self._updateVariables() return 2 def fontfacerule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSFontFaceRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: self.insertRule(rule) return 3 def mediarule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSMediaRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: self.insertRule(rule) return 3 def pagerule(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSPageRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: self.insertRule(rule) return 3 def unknownrule(expected, seq, token, tokenizer): # parse and consume tokens in any case if token[1] in cssutils.css.MarginRule.margins: self._log.error(u'CSSStylesheet: MarginRule out CSSPageRule.', token, neverraise=True) rule = cssutils.css.MarginRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) else: self._log.warn(u'CSSStylesheet: Unknown @rule found.', token, neverraise=True) rule = cssutils.css.CSSUnknownRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: self.insertRule(rule) # or 0 for py3 return max(1, expected or 0) def ruleset(expected, seq, token, tokenizer): # parse and consume tokens in any case rule = cssutils.css.CSSStyleRule(parentStyleSheet=self) rule.cssText = self._tokensupto2(tokenizer, token) if rule.wellformed: self.insertRule(rule) return 3 # save for possible reset oldCssRules = self.cssRules oldNamespaces = self._namespaces self.cssRules = cssutils.css.CSSRuleList() # simple during parse self._namespaces = namespaces self._variables = CSSVariablesDeclaration() # not used?! newseq = [] # ['CHARSET', 'IMPORT', ('VAR', NAMESPACE'), ('PAGE', 'MEDIA', ruleset)] wellformed, expected = self._parse(0, newseq, tokenizer, {'S': S, 'COMMENT': COMMENT, 'CDO': lambda *ignored: None, 'CDC': lambda *ignored: None, 'CHARSET_SYM': charsetrule, 'FONT_FACE_SYM': fontfacerule, 'IMPORT_SYM': importrule, 'NAMESPACE_SYM': namespacerule, 'PAGE_SYM': pagerule, 'MEDIA_SYM': mediarule, 'VARIABLES_SYM': variablesrule, 'ATKEYWORD': unknownrule }, default=ruleset) if wellformed: # use proper namespace object self._namespaces = _Namespaces(parentStyleSheet=self, log=self._log) self._cleanNamespaces() else: # reset self._cssRules = oldCssRules self._namespaces = oldNamespaces self._updateVariables() self._cleanNamespaces() cssText = property(_getCssText, _setCssText, "Textual representation of the stylesheet (a byte string)") def _resolveImport(self, url): """Read (encoding, enctype, decodedContent) from `url` for @import sheets.""" try: # only available during parsing of a complete sheet parentEncoding = self.__newEncoding except AttributeError: try: # explicit @charset parentEncoding = self._cssRules[0].encoding except (IndexError, AttributeError): # default not UTF-8 but None! parentEncoding = None return _readUrl(url, fetcher=self._fetcher, overrideEncoding=self.__encodingOverride, parentEncoding=parentEncoding) def _setCssTextWithEncodingOverride(self, cssText, encodingOverride=None, encoding=None): """Set `cssText` but use `encodingOverride` to overwrite detected encoding. This is used by parse and @import during setting of cssText. If `encoding` is given use this but do not save as `encodingOverride`. """ if encodingOverride: # encoding during resolving of @import self.__encodingOverride = encodingOverride if encoding: # save for nested @import self.__newEncoding = encoding self.cssText = cssText if encodingOverride: # set encodingOverride explicit again! self.encoding = self.__encodingOverride # del? self.__encodingOverride = None elif encoding: # may e.g. be httpEncoding self.encoding = encoding try: del self.__newEncoding except AttributeError, e: pass def _setFetcher(self, fetcher=None): """Set @import URL loader, if None the default is used.""" self._fetcher = fetcher def _getEncoding(self): """Encoding set in :class:`~cssutils.css.CSSCharsetRule` or if ``None`` resulting in default ``utf-8`` encoding being used.""" try: return self._cssRules[0].encoding except (IndexError, AttributeError): return 'utf-8' def _setEncoding(self, encoding): """Set `encoding` of charset rule if present in sheet or insert a new :class:`~cssutils.css.CSSCharsetRule` with given `encoding`. If `encoding` is None removes charsetrule if present resulting in default encoding of utf-8. """ try: rule = self._cssRules[0] except IndexError: rule = None if rule and rule.CHARSET_RULE == rule.type: if encoding: rule.encoding = encoding else: self.deleteRule(0) elif encoding: self.insertRule(cssutils.css.CSSCharsetRule(encoding=encoding), 0) encoding = property(_getEncoding, _setEncoding, "(cssutils) Reflect encoding of an @charset rule or 'utf-8' " "(default) if set to ``None``") namespaces = property(lambda self: self._namespaces, doc="All Namespaces used in this CSSStyleSheet.") def _updateVariables(self): """Updates self._variables, called when @import or @variables rules is added to sheet. """ for r in self.cssRules.rulesOfType(CSSRule.IMPORT_RULE): s = r.styleSheet if s: for var in s.variables: self._variables.setVariable(var, s.variables[var]) # for r in self.cssRules.rulesOfType(CSSRule.IMPORT_RULE): # for vr in r.styleSheet.cssRules.rulesOfType(CSSRule.VARIABLES_RULE): # for var in vr.variables: # self._variables.setVariable(var, vr.variables[var]) for vr in self.cssRules.rulesOfType(CSSRule.VARIABLES_RULE): for var in vr.variables: self._variables.setVariable(var, vr.variables[var]) variables = property(lambda self: self._variables, doc=u"A :class:`cssutils.css.CSSVariablesDeclaration` " u"containing all available variables in this " u"CSSStyleSheet including the ones defined in " u"imported sheets.") def add(self, rule): """Add `rule` to style sheet at appropriate position. Same as ``insertRule(rule, inOrder=True)``. """ return self.insertRule(rule, index=None, inOrder=True) def deleteRule(self, index): """Delete rule at `index` from the style sheet. :param index: The `index` of the rule to be removed from the StyleSheet's rule list. For an `index` < 0 **no** :exc:`~xml.dom.IndexSizeErr` is raised but rules for normal Python lists are used. E.g. ``deleteRule(-1)`` removes the last rule in cssRules. `index` may also be a CSSRule object which will then be removed from the StyleSheet. :exceptions: - :exc:`~xml.dom.IndexSizeErr`: Raised if the specified index does not correspond to a rule in the style sheet's rule list. - :exc:`~xml.dom.NamespaceErr`: Raised if removing this rule would result in an invalid StyleSheet - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this style sheet is readonly. """ self._checkReadonly() if isinstance(index, CSSRule): for i, r in enumerate(self.cssRules): if index == r: index = i break else: raise xml.dom.IndexSizeErr(u"CSSStyleSheet: Not a rule in" " this sheets'a cssRules list: %s" % index) try: rule = self._cssRules[index] except IndexError: raise xml.dom.IndexSizeErr( u'CSSStyleSheet: %s is not a valid index in the rulelist of ' u'length %i' % (index, self._cssRules.length)) else: if rule.type == rule.NAMESPACE_RULE: # check all namespacerules if used uris = [r.namespaceURI for r in self if r.type == r.NAMESPACE_RULE] useduris = self._getUsedURIs() if rule.namespaceURI in useduris and\ uris.count(rule.namespaceURI) == 1: raise xml.dom.NoModificationAllowedErr( u'CSSStyleSheet: NamespaceURI defined in this rule is ' u'used, cannot remove.') return rule._parentStyleSheet = None # detach del self._cssRules[index] # delete from StyleSheet def insertRule(self, rule, index=None, inOrder=False, _clean=True): """ Used to insert a new rule into the style sheet. The new rule now becomes part of the cascade. :param rule: a parsable DOMString, in cssutils also a :class:`~cssutils.css.CSSRule` or :class:`~cssutils.css.CSSRuleList` :param index: of the rule before the new rule will be inserted. If the specified `index` is equal to the length of the StyleSheet's rule collection, the rule will be added to the end of the style sheet. If `index` is not given or ``None`` rule will be appended to rule list. :param inOrder: if ``True`` the rule will be put to a proper location while ignoring `index` and without raising :exc:`~xml.dom.HierarchyRequestErr`. The resulting index is returned nevertheless. :returns: The index within the style sheet's rule collection :Exceptions: - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at the specified `index` e.g. if an @import rule is inserted after a standard rule set or other at-rule. - :exc:`~xml.dom.IndexSizeErr`: Raised if the specified `index` is not a valid insertion point. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this style sheet is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified rule has a syntax error and is unparsable. """ self._checkReadonly() # check position if index is None: index = len(self._cssRules) elif index < 0 or index > self._cssRules.length: raise xml.dom.IndexSizeErr( u'CSSStyleSheet: Invalid index %s for CSSRuleList with a ' u'length of %s.' % (index, self._cssRules.length)) return if isinstance(rule, basestring): # init a temp sheet which has the same properties as self tempsheet = CSSStyleSheet(href=self.href, media=self.media, title=self.title, parentStyleSheet=self.parentStyleSheet, ownerRule=self.ownerRule) tempsheet._ownerNode = self.ownerNode tempsheet._fetcher = self._fetcher # prepend encoding if in this sheet to be able to use it in # @import rules encoding resolution # do not add if new rule startswith "@charset" (which is exact!) if not rule.startswith(u'@charset') and (self._cssRules and self._cssRules[0].type == self._cssRules[0].CHARSET_RULE): # rule 0 is @charset! newrulescount, newruleindex = 2, 1 rule = self._cssRules[0].cssText + rule else: newrulescount, newruleindex = 1, 0 # parse the new rule(s) tempsheet.cssText = (rule, self._namespaces) if len(tempsheet.cssRules) != newrulescount or (not isinstance( tempsheet.cssRules[newruleindex], cssutils.css.CSSRule)): self._log.error(u'CSSStyleSheet: Not a CSSRule: %s' % rule) return rule = tempsheet.cssRules[newruleindex] rule._parentStyleSheet = None # done later? # TODO: #tempsheet._namespaces = self._namespaces #variables? elif isinstance(rule, cssutils.css.CSSRuleList): # insert all rules for i, r in enumerate(rule): self.insertRule(r, index + i) return index if not rule.wellformed: self._log.error(u'CSSStyleSheet: Invalid rules cannot be added.') return # CHECK HIERARCHY # @charset if rule.type == rule.CHARSET_RULE: if inOrder: index = 0 # always first and only if (self._cssRules and self._cssRules[0].type == rule.CHARSET_RULE): self._cssRules[0].encoding = rule.encoding else: self._cssRules.insert(0, rule) elif index != 0 or (self._cssRules and self._cssRules[0].type == rule.CHARSET_RULE): self._log.error( u'CSSStylesheet: @charset only allowed once at the' ' beginning of a stylesheet.', error=xml.dom.HierarchyRequestErr) return else: self._cssRules.insert(index, rule) # @unknown or comment elif rule.type in (rule.UNKNOWN_RULE, rule.COMMENT) and not inOrder: if index == 0 and self._cssRules and\ self._cssRules[0].type == rule.CHARSET_RULE: self._log.error( u'CSSStylesheet: @charset must be the first rule.', error=xml.dom.HierarchyRequestErr) return else: self._cssRules.insert(index, rule) # @import elif rule.type == rule.IMPORT_RULE: if inOrder: # automatic order if rule.type in (r.type for r in self): # find last of this type for i, r in enumerate(reversed(self._cssRules)): if r.type == rule.type: index = len(self._cssRules) - i break else: # find first point to insert if self._cssRules and\ self._cssRules[0].type in (rule.CHARSET_RULE, rule.COMMENT): index = 1 else: index = 0 else: # after @charset if index == 0 and self._cssRules and\ self._cssRules[0].type == rule.CHARSET_RULE: self._log.error( u'CSSStylesheet: Found @charset at index 0.', error=xml.dom.HierarchyRequestErr) return # before @namespace @variables @page @font-face @media stylerule for r in self._cssRules[:index]: if r.type in (r.NAMESPACE_RULE, r.VARIABLES_RULE, r.MEDIA_RULE, r.PAGE_RULE, r.STYLE_RULE, r.FONT_FACE_RULE): self._log.error( u'CSSStylesheet: Cannot insert @import here,' ' found @namespace, @variables, @media, @page or' ' CSSStyleRule before index %s.' % index, error=xml.dom.HierarchyRequestErr) return self._cssRules.insert(index, rule) self._updateVariables() # @namespace elif rule.type == rule.NAMESPACE_RULE: if inOrder: if rule.type in (r.type for r in self): # find last of this type for i, r in enumerate(reversed(self._cssRules)): if r.type == rule.type: index = len(self._cssRules) - i break else: # find first point to insert for i, r in enumerate(self._cssRules): if r.type in (r.VARIABLES_RULE, r.MEDIA_RULE, r.PAGE_RULE, r.STYLE_RULE, r.FONT_FACE_RULE, r.UNKNOWN_RULE, r.COMMENT): index = i # before these break else: # after @charset and @import for r in self._cssRules[index:]: if r.type in (r.CHARSET_RULE, r.IMPORT_RULE): self._log.error( u'CSSStylesheet: Cannot insert @namespace here,' ' found @charset or @import after index %s.' % index, error=xml.dom.HierarchyRequestErr) return # before @variables @media @page @font-face and stylerule for r in self._cssRules[:index]: if r.type in (r.VARIABLES_RULE, r.MEDIA_RULE, r.PAGE_RULE, r.STYLE_RULE, r.FONT_FACE_RULE): self._log.error( u'CSSStylesheet: Cannot insert @namespace here,' ' found @variables, @media, @page or CSSStyleRule' ' before index %s.' % index, error=xml.dom.HierarchyRequestErr) return if not (rule.prefix in self.namespaces and self.namespaces[rule.prefix] == rule.namespaceURI): # no doublettes self._cssRules.insert(index, rule) if _clean: self._cleanNamespaces() # @variables elif rule.type == rule.VARIABLES_RULE: if inOrder: if rule.type in (r.type for r in self): # find last of this type for i, r in enumerate(reversed(self._cssRules)): if r.type == rule.type: index = len(self._cssRules) - i break else: # find first point to insert for i, r in enumerate(self._cssRules): if r.type in (r.MEDIA_RULE, r.PAGE_RULE, r.STYLE_RULE, r.FONT_FACE_RULE, r.UNKNOWN_RULE, r.COMMENT): index = i # before these break else: # after @charset @import @namespace for r in self._cssRules[index:]: if r.type in (r.CHARSET_RULE, r.IMPORT_RULE, r.NAMESPACE_RULE): self._log.error( u'CSSStylesheet: Cannot insert @variables here,' ' found @charset, @import or @namespace after' ' index %s.' % index, error=xml.dom.HierarchyRequestErr) return # before @media @page @font-face and stylerule for r in self._cssRules[:index]: if r.type in (r.MEDIA_RULE, r.PAGE_RULE, r.STYLE_RULE, r.FONT_FACE_RULE): self._log.error( u'CSSStylesheet: Cannot insert @variables here,' ' found @media, @page or CSSStyleRule' ' before index %s.' % index, error=xml.dom.HierarchyRequestErr) return self._cssRules.insert(index, rule) self._updateVariables() # all other where order is not important else: if inOrder: # simply add to end as no specific order self._cssRules.append(rule) index = len(self._cssRules) - 1 else: for r in self._cssRules[index:]: if r.type in (r.CHARSET_RULE, r.IMPORT_RULE, r.NAMESPACE_RULE): self._log.error( u'CSSStylesheet: Cannot insert rule here, found ' u'@charset, @import or @namespace before index %s.' % index, error=xml.dom.HierarchyRequestErr) return self._cssRules.insert(index, rule) # post settings rule._parentStyleSheet = self if rule.IMPORT_RULE == rule.type and not rule.hrefFound: # try loading the imported sheet which has new relative href now rule.href = rule.href return index ownerRule = property(lambda self: self._ownerRule, doc=u'A ref to an @import rule if it is imported, ' u'else ``None``.') @Deprecated(u'Use ``cssutils.setSerializer(serializer)`` instead.') def setSerializer(self, cssserializer): """Set the cssutils global Serializer used for all output.""" if isinstance(cssserializer, cssutils.CSSSerializer): cssutils.ser = cssserializer else: raise ValueError(u'Serializer must be an instance of ' u'cssutils.CSSSerializer.') @Deprecated(u'Set pref in ``cssutils.ser.prefs`` instead.') def setSerializerPref(self, pref, value): """Set a Preference of CSSSerializer used for output. See :class:`cssutils.serialize.Preferences` for possible preferences to be set. """ cssutils.ser.prefs.__setattr__(pref, value) cssutils-1.0/src/cssutils/css/cssimportrule.py0000666000175000017500000003540712126054542020444 0ustar hugohugo"""CSSImportRule implements DOM Level 2 CSS CSSImportRule plus the ``name`` property from http://www.w3.org/TR/css3-cascade/#cascading.""" __all__ = ['CSSImportRule'] __docformat__ = 'restructuredtext' __version__ = '$Id$' import cssrule import cssutils import os import urlparse import xml.dom class CSSImportRule(cssrule.CSSRule): """ Represents an @import rule within a CSS style sheet. The @import rule is used to import style rules from other style sheets. Format:: import : IMPORT_SYM S* [STRING|URI] S* [ medium [ COMMA S* medium]* ]? S* STRING? S* ';' S* ; """ def __init__(self, href=None, mediaText=None, name=None, parentRule=None, parentStyleSheet=None, readonly=False): """ If readonly allows setting of properties in constructor only :param href: location of the style sheet to be imported. :param mediaText: A list of media types for which this style sheet may be used as a string :param name: Additional name of imported style sheet """ super(CSSImportRule, self).__init__(parentRule=parentRule, parentStyleSheet=parentStyleSheet) self._atkeyword = u'@import' self._styleSheet = None # string or uri used for reserialization self.hreftype = None # prepare seq seq = self._tempSeq() seq.append(None, 'href') #seq.append(None, 'media') seq.append(None, 'name') self._setSeq(seq) # 1. media if mediaText: self.media = mediaText else: # must be all for @import self.media = cssutils.stylesheets.MediaList(mediaText=u'all') # 2. name self.name = name # 3. href and styleSheet self.href = href self._readonly = readonly def __repr__(self): if self._usemedia: mediaText = self.media.mediaText else: mediaText = None return u"cssutils.css.%s(href=%r, mediaText=%r, name=%r)" % ( self.__class__.__name__, self.href, self.media.mediaText, self.name) def __str__(self): if self._usemedia: mediaText = self.media.mediaText else: mediaText = None return u""\ % (self.__class__.__name__, self.href, mediaText, self.name, id(self)) _usemedia = property(lambda self: self.media.mediaText not in (u'', u'all'), doc="if self.media is used (or simply empty)") def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_CSSImportRule(self) def _setCssText(self, cssText): """ :exceptions: - :exc:`~xml.dom.HierarchyRequestErr`: Raised if the rule cannot be inserted at this point in the style sheet. - :exc:`~xml.dom.InvalidModificationErr`: Raised if the specified CSS string value represents a different type of rule than the current one. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if the rule is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. """ super(CSSImportRule, self)._setCssText(cssText) tokenizer = self._tokenize2(cssText) attoken = self._nexttoken(tokenizer, None) if self._type(attoken) != self._prods.IMPORT_SYM: self._log.error(u'CSSImportRule: No CSSImportRule found: %s' % self._valuestr(cssText), error=xml.dom.InvalidModificationErr) else: # for closures: must be a mutable new = {'keyword': self._tokenvalue(attoken), 'href': None, 'hreftype': None, 'media': None, 'name': None, 'wellformed': True } def __doname(seq, token): # called by _string or _ident new['name'] = self._stringtokenvalue(token) seq.append(new['name'], 'name') return ';' def _string(expected, seq, token, tokenizer=None): if 'href' == expected: # href new['href'] = self._stringtokenvalue(token) new['hreftype'] = 'string' seq.append(new['href'], 'href') return 'media name ;' elif 'name' in expected: # name return __doname(seq, token) else: new['wellformed'] = False self._log.error( u'CSSImportRule: Unexpected string.', token) return expected def _uri(expected, seq, token, tokenizer=None): # href if 'href' == expected: uri = self._uritokenvalue(token) new['hreftype'] = 'uri' new['href'] = uri seq.append(new['href'], 'href') return 'media name ;' else: new['wellformed'] = False self._log.error( u'CSSImportRule: Unexpected URI.', token) return expected def _ident(expected, seq, token, tokenizer=None): # medialist ending with ; which is checked upon too if expected.startswith('media'): mediatokens = self._tokensupto2( tokenizer, importmediaqueryendonly=True) mediatokens.insert(0, token) # push found token last = mediatokens.pop() # retrieve ; lastval, lasttyp = self._tokenvalue(last), self._type(last) if lastval != u';' and lasttyp not in ('EOF', self._prods.STRING): new['wellformed'] = False self._log.error(u'CSSImportRule: No ";" found: %s' % self._valuestr(cssText), token=token) newMedia = cssutils.stylesheets.MediaList(parentRule=self) newMedia.mediaText = mediatokens if newMedia.wellformed: new['media'] = newMedia seq.append(newMedia, 'media') else: new['wellformed'] = False self._log.error(u'CSSImportRule: Invalid MediaList: %s' % self._valuestr(cssText), token=token) if lasttyp == self._prods.STRING: # name return __doname(seq, last) else: return 'EOF' # ';' is token "last" else: new['wellformed'] = False self._log.error(u'CSSImportRule: Unexpected ident.', token) return expected def _char(expected, seq, token, tokenizer=None): # final ; val = self._tokenvalue(token) if expected.endswith(';') and u';' == val: return 'EOF' else: new['wellformed'] = False self._log.error( u'CSSImportRule: Unexpected char.', token) return expected # import : IMPORT_SYM S* [STRING|URI] # S* [ medium [ ',' S* medium]* ]? ';' S* # STRING? # see http://www.w3.org/TR/css3-cascade/#cascading # ; newseq = self._tempSeq() wellformed, expected = self._parse(expected='href', seq=newseq, tokenizer=tokenizer, productions={'STRING': _string, 'URI': _uri, 'IDENT': _ident, 'CHAR': _char}, new=new) # wellformed set by parse ok = wellformed and new['wellformed'] # post conditions if not new['href']: ok = False self._log.error(u'CSSImportRule: No href found: %s' % self._valuestr(cssText)) if expected != 'EOF': ok = False self._log.error(u'CSSImportRule: No ";" found: %s' % self._valuestr(cssText)) # set all if ok: self._setSeq(newseq) self.atkeyword = new['keyword'] self.hreftype = new['hreftype'] self.name = new['name'] if new['media']: self.media = new['media'] else: # must be all for @import self.media = cssutils.stylesheets.MediaList(mediaText=u'all') # needs new self.media self.href = new['href'] cssText = property(fget=_getCssText, fset=_setCssText, doc="(DOM) The parsable textual representation of this rule.") def _setHref(self, href): # set new href self._href = href # update seq for i, item in enumerate(self.seq): val, type_ = item.value, item.type if 'href' == type_: self._seq[i] = (href, type_, item.line, item.col) break importedSheet = cssutils.css.CSSStyleSheet(media=self.media, ownerRule=self, title=self.name) self.hrefFound = False # set styleSheet if href and self.parentStyleSheet: # loading errors are all catched! # relative href parentHref = self.parentStyleSheet.href if parentHref is None: # use cwd instead parentHref = cssutils.helper.path2url(os.getcwd()) + '/' fullhref = urlparse.urljoin(parentHref, self.href) # all possible exceptions are ignored try: usedEncoding, enctype, cssText = \ self.parentStyleSheet._resolveImport(fullhref) if cssText is None: # catched in next except below! raise IOError('Cannot read Stylesheet.') # contentEncoding with parentStyleSheet.overrideEncoding, # HTTP or parent encodingOverride, encoding = None, None if enctype == 0: encodingOverride = usedEncoding elif 0 < enctype < 5: encoding = usedEncoding # inherit fetcher for @imports in styleSheet importedSheet._href = fullhref importedSheet._setFetcher(self.parentStyleSheet._fetcher) importedSheet._setCssTextWithEncodingOverride( cssText, encodingOverride=encodingOverride, encoding=encoding) except (OSError, IOError, ValueError), e: self._log.warn(u'CSSImportRule: While processing imported ' u'style sheet href=%s: %r' % (self.href, e), neverraise=True) else: # used by resolveImports if to keep unprocessed href self.hrefFound = True self._styleSheet = importedSheet _href = None # needs to be set href = property(lambda self: self._href, _setHref, doc=u"Location of the style sheet to be imported.") def _setMedia(self, media): """ :param media: a :class:`~cssutils.stylesheets.MediaList` or string """ self._checkReadonly() if isinstance(media, basestring): self._media = cssutils.stylesheets.MediaList(mediaText=media, parentRule=self) else: media._parentRule = self self._media = media # update seq ihref = 0 for i, item in enumerate(self.seq): if item.type == 'href': ihref = i elif item.type == 'media': self.seq[i] = (self._media, 'media', None, None) break else: # if no media until now add after href self.seq.insert(ihref+1, self._media, 'media', None, None) media = property(lambda self: self._media, _setMedia, doc=u"(DOM) A list of media types for this rule " u"of type :class:`~cssutils.stylesheets.MediaList`.") def _setName(self, name=u''): """Raises xml.dom.SyntaxErr if name is not a string.""" if name is None or isinstance(name, basestring): # "" or '' handled as None if not name: name = None # save name self._name = name # update seq for i, item in enumerate(self.seq): val, typ = item.value, item.type if 'name' == typ: self._seq[i] = (name, typ, item.line, item.col) break # set title of imported sheet if self.styleSheet: self.styleSheet.title = name else: self._log.error(u'CSSImportRule: Not a valid name: %s' % name) name = property(lambda self: self._name, _setName, doc=u"An optional name for the imported sheet.") styleSheet = property(lambda self: self._styleSheet, doc=u"(readonly) The style sheet referred to by this " u"rule.") type = property(lambda self: self.IMPORT_RULE, doc=u"The type of this rule, as defined by a CSSRule " u"type constant.") def _getWellformed(self): "Depending on if media is used at all." if self._usemedia: return bool(self.href and self.media.wellformed) else: return bool(self.href) wellformed = property(_getWellformed) cssutils-1.0/src/cssutils/css/cssvariablesdeclaration.py0000666000175000017500000003053512253143234022413 0ustar hugohugo"""CSSVariablesDeclaration http://disruptive-innovations.com/zoo/cssvariables/#mozTocId496530 """ __all__ = ['CSSVariablesDeclaration'] __docformat__ = 'restructuredtext' __version__ = '$Id: cssstyledeclaration.py 1819 2009-08-01 20:52:43Z cthedot $' from cssutils.prodparser import * from cssutils.helper import normalize from value import PropertyValue import cssutils import itertools import xml.dom class CSSVariablesDeclaration(cssutils.util._NewBase): """The CSSVariablesDeclaration interface represents a single block of variable declarations. """ def __init__(self, cssText=u'', parentRule=None, readonly=False): """ :param cssText: Shortcut, sets CSSVariablesDeclaration.cssText :param parentRule: The CSS rule that contains this declaration block or None if this CSSVariablesDeclaration is not attached to a CSSRule. :param readonly: defaults to False Format:: variableset : vardeclaration [ ';' S* vardeclaration ]* S* ; vardeclaration : varname ':' S* term ; varname : IDENT S* ; """ super(CSSVariablesDeclaration, self).__init__() self._parentRule = parentRule self._vars = {} if cssText: self.cssText = cssText self._readonly = readonly def __repr__(self): return u"cssutils.css.%s(cssText=%r)" % (self.__class__.__name__, self.cssText) def __str__(self): return u"" % ( self.__class__.__name__, self.length, id(self)) def __contains__(self, variableName): """Check if a variable is in variable declaration block. :param variableName: a string """ return normalize(variableName) in self.keys() def __getitem__(self, variableName): """Retrieve the value of variable ``variableName`` from this declaration. """ return self.getVariableValue(variableName) def __setitem__(self, variableName, value): self.setVariable(variableName, value) def __delitem__(self, variableName): return self.removeVariable(variableName) def __iter__(self): """Iterator of names of set variables.""" for name in self.keys(): yield name def keys(self): """Analoguous to standard dict returns variable names which are set in this declaration.""" return self._vars.keys() def _getCssText(self): """Return serialized property cssText.""" return cssutils.ser.do_css_CSSVariablesDeclaration(self) def _setCssText(self, cssText): """Setting this attribute will result in the parsing of the new value and resetting of all the properties in the declaration block including the removal or addition of properties. :exceptions: - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this declaration is readonly or a property is readonly. - :exc:`~xml.dom.SyntaxErr`: Raised if the specified CSS string value has a syntax error and is unparsable. Format:: variableset : vardeclaration [ ';' S* vardeclaration ]* ; vardeclaration : varname ':' S* term ; varname : IDENT S* ; expr : [ VARCALL | term ] [ operator [ VARCALL | term ] ]* ; """ self._checkReadonly() vardeclaration = Sequence( PreDef.ident(), PreDef.char(u':', u':', toSeq=False, optional=True), #PreDef.S(toSeq=False, optional=True), Prod(name=u'term', match=lambda t, v: True, toSeq=lambda t, tokens: (u'value', PropertyValue(itertools.chain([t], tokens), parent=self) ) ) ) prods = Sequence(vardeclaration, Sequence(PreDef.S(optional=True), PreDef.char(u';', u';', toSeq=False, optional=True), PreDef.S(optional=True), vardeclaration, minmax=lambda: (0, None)), PreDef.S(optional=True), PreDef.char(u';', u';', toSeq=False, optional=True) ) # parse wellformed, seq, store, notused = \ ProdParser().parse(cssText, u'CSSVariableDeclaration', prods, emptyOk=True) if wellformed: newseq = self._tempSeq() newvars = {} # seq contains only name: value pairs plus comments etc nameitem = None for item in seq: if u'IDENT' == item.type: nameitem = item elif u'value' == item.type: nname = normalize(nameitem.value) if nname in newvars: # replace var with same name for i, it in enumerate(newseq): if normalize(it.value[0]) == nname: newseq.replace(i, (nameitem.value, item.value), 'var', nameitem.line, nameitem.col) else: # saved non normalized name for reserialization newseq.append((nameitem.value, item.value), 'var', nameitem.line, nameitem.col) # newseq.append((nameitem.value, item.value), # 'var', # nameitem.line, nameitem.col) newvars[nname] = item.value else: newseq.appendItem(item) self._setSeq(newseq) self._vars = newvars self.wellformed = True cssText = property(_getCssText, _setCssText, doc=u"(DOM) A parsable textual representation of the declaration " u"block excluding the surrounding curly braces.") def _setParentRule(self, parentRule): self._parentRule = parentRule parentRule = property(lambda self: self._parentRule, _setParentRule, doc=u"(DOM) The CSS rule that contains this" u" declaration block or None if this block" u" is not attached to a CSSRule.") def getVariableValue(self, variableName): """Used to retrieve the value of a variable if it has been explicitly set within this variable declaration block. :param variableName: The name of the variable. :returns: the value of the variable if it has been explicitly set in this variable declaration block. Returns the empty string if the variable has not been set. """ try: return self._vars[normalize(variableName)].cssText except KeyError, e: return u'' def removeVariable(self, variableName): """Used to remove a variable if it has been explicitly set within this variable declaration block. :param variableName: The name of the variable. :returns: the value of the variable if it has been explicitly set for this variable declaration block. Returns the empty string if the variable has not been set. :exceptions: - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this declaration is readonly is readonly. """ normalname = variableName try: r = self._vars[normalname] except KeyError, e: return u'' else: self.seq._readonly = False if normalname in self._vars: for i, x in enumerate(self.seq): if x.value[0] == variableName: del self.seq[i] self.seq._readonly = True del self._vars[normalname] return r.cssText def setVariable(self, variableName, value): """Used to set a variable value within this variable declaration block. :param variableName: The name of the CSS variable. :param value: The new value of the variable, may also be a PropertyValue object. :exceptions: - :exc:`~xml.dom.SyntaxErr`: Raised if the specified value has a syntax error and is unparsable. - :exc:`~xml.dom.NoModificationAllowedErr`: Raised if this declaration is readonly or the property is readonly. """ self._checkReadonly() # check name wellformed, seq, store, unused = \ ProdParser().parse(normalize(variableName), u'variableName', Sequence(PreDef.ident())) if not wellformed: self._log.error(u'Invalid variableName: %r: %r' % (variableName, value)) else: # check value if isinstance(value, PropertyValue): v = value else: v = PropertyValue(cssText=value, parent=self) if not v.wellformed: self._log.error(u'Invalid variable value: %r: %r' % (variableName, value)) else: # update seq self.seq._readonly = False variableName = normalize(variableName) if variableName in self._vars: for i, x in enumerate(self.seq): if x.value[0] == variableName: self.seq.replace(i, [variableName, v], x.type, x.line, x.col) break else: self.seq.append([variableName, v], 'var') self.seq._readonly = True self._vars[variableName] = v def item(self, index): """Used to retrieve the variables that have been explicitly set in this variable declaration block. The order of the variables retrieved using this method does not have to be the order in which they were set. This method can be used to iterate over all variables in this variable declaration block. :param index: of the variable name to retrieve, negative values behave like negative indexes on Python lists, so -1 is the last element :returns: The name of the variable at this ordinal position. The empty string if no variable exists at this position. """ try: return self.keys()[index] except IndexError: return u'' length = property(lambda self: len(self._vars), doc=u"The number of variables that have been explicitly set in this" u" variable declaration block. The range of valid indices is 0" u" to length-1 inclusive.") cssutils-1.0/src/cssutils/script.py0000666000175000017500000003124012126054542016234 0ustar hugohugo"""classes and functions used by cssutils scripts """ __all__ = ['CSSCapture', 'csscombine'] __docformat__ = 'restructuredtext' __version__ = '$Id: parse.py 1323 2008-07-06 18:13:57Z cthedot $' import HTMLParser import codecs import cssutils import errno import logging import os import sys import urllib2 import urlparse try: import cssutils.encutils as encutils except ImportError: try: import encutils except ImportError: sys.exit("You need encutils from http://cthedot.de/encutils/") # types of sheets in HTML LINK = 0 # STYLE = 1 # class CSSCaptureHTMLParser(HTMLParser.HTMLParser): """CSSCapture helper: Parse given data for link and style elements""" curtag = u'' sheets = [] # (type, [atts, cssText]) def _loweratts(self, atts): return dict([(a.lower(), v.lower()) for a, v in atts]) def handle_starttag(self, tag, atts): if tag == u'link': atts = self._loweratts(atts) if u'text/css' == atts.get(u'type', u''): self.sheets.append((LINK, atts)) elif tag == u'style': # also get content of style atts = self._loweratts(atts) if u'text/css' == atts.get(u'type', u''): self.sheets.append((STYLE, [atts, u''])) self.curtag = tag else: # close as only intersting