diff options
| author | terminaldweller <thabogre@gmail.com> | 2022-04-22 05:25:23 +0000 | 
|---|---|---|
| committer | terminaldweller <thabogre@gmail.com> | 2022-04-22 05:25:23 +0000 | 
| commit | 641062ef0fca969feae3980afd0d22e436a26c41 (patch) | |
| tree | 5a882d43a56662f82b805e7943873211c4a5b6a1 | |
| parent | Initial commit (diff) | |
| download | tabbed-main.tar.gz tabbed-main.zip | |
Diffstat (limited to '')
| -rw-r--r-- | LICENSE | 697 | ||||
| -rw-r--r-- | Makefile | 60 | ||||
| -rw-r--r-- | README | 22 | ||||
| -rw-r--r-- | TODO | 4 | ||||
| -rw-r--r-- | arg.h | 52 | ||||
| -rw-r--r-- | config.def.h | 74 | ||||
| -rw-r--r-- | config.h | 74 | ||||
| -rw-r--r-- | config.mk | 25 | ||||
| -rw-r--r-- | tabbed-0.6-xft.diff | 19 | ||||
| -rw-r--r-- | tabbed-clientnumber-0.6.diff | 23 | ||||
| -rw-r--r-- | tabbed-xresources-20210317-dabf6a2.diff | 15 | ||||
| -rw-r--r-- | tabbed.1 | 144 | ||||
| -rw-r--r-- | tabbed.c | 1261 | 
13 files changed, 1796 insertions, 674 deletions
| @@ -1,674 +1,23 @@ -                    GNU GENERAL PUBLIC LICENSE -                       Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - -                            Preamble - -  The GNU General Public License is a free, copyleft license for -software and other kinds of works. - -  The licenses for most software and other practical works are designed -to take away your freedom to share and change the works.  By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users.  We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors.  You can apply it to -your programs, too. - -  When we speak of free software, we are referring to freedom, not -price.  Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - -  To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights.  Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - -  For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received.  You must make sure that they, too, receive -or can get the source code.  And you must show them these terms so they -know their rights. - -  Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - -  For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software.  For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - -  Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so.  This is fundamentally incompatible with the aim of -protecting users' freedom to change the software.  The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable.  Therefore, we -have designed this version of the GPL to prohibit the practice for those -products.  If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - -  Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary.  To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - -  The precise terms and conditions for copying, distribution and -modification follow. - -                       TERMS AND CONDITIONS - -  0. Definitions. - -  "This License" refers to version 3 of the GNU General Public License. - -  "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - -  "The Program" refers to any copyrightable work licensed under this -License.  Each licensee is addressed as "you".  "Licensees" and -"recipients" may be individuals or organizations. - -  To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy.  The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - -  A "covered work" means either the unmodified Program or a work based -on the Program. - -  To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy.  Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - -  To "convey" a work means any kind of propagation that enables other -parties to make or receive copies.  Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - -  An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License.  If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - -  1. Source Code. - -  The "source code" for a work means the preferred form of the work -for making modifications to it.  "Object code" means any non-source -form of a work. - -  A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - -  The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form.  A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - -  The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities.  However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work.  For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - -  The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - -  The Corresponding Source for a work in source code form is that -same work. - -  2. Basic Permissions. - -  All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met.  This License explicitly affirms your unlimited -permission to run the unmodified Program.  The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work.  This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - -  You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force.  You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright.  Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - -  Conveying under any other circumstances is permitted solely under -the conditions stated below.  Sublicensing is not allowed; section 10 -makes it unnecessary. - -  3. Protecting Users' Legal Rights From Anti-Circumvention Law. - -  No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - -  When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - -  4. Conveying Verbatim Copies. - -  You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - -  You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - -  5. Conveying Modified Source Versions. - -  You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - -    a) The work must carry prominent notices stating that you modified -    it, and giving a relevant date. - -    b) The work must carry prominent notices stating that it is -    released under this License and any conditions added under section -    7.  This requirement modifies the requirement in section 4 to -    "keep intact all notices". - -    c) You must license the entire work, as a whole, under this -    License to anyone who comes into possession of a copy.  This -    License will therefore apply, along with any applicable section 7 -    additional terms, to the whole of the work, and all its parts, -    regardless of how they are packaged.  This License gives no -    permission to license the work in any other way, but it does not -    invalidate such permission if you have separately received it. - -    d) If the work has interactive user interfaces, each must display -    Appropriate Legal Notices; however, if the Program has interactive -    interfaces that do not display Appropriate Legal Notices, your -    work need not make them do so. - -  A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit.  Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - -  6. Conveying Non-Source Forms. - -  You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - -    a) Convey the object code in, or embodied in, a physical product -    (including a physical distribution medium), accompanied by the -    Corresponding Source fixed on a durable physical medium -    customarily used for software interchange. - -    b) Convey the object code in, or embodied in, a physical product -    (including a physical distribution medium), accompanied by a -    written offer, valid for at least three years and valid for as -    long as you offer spare parts or customer support for that product -    model, to give anyone who possesses the object code either (1) a -    copy of the Corresponding Source for all the software in the -    product that is covered by this License, on a durable physical -    medium customarily used for software interchange, for a price no -    more than your reasonable cost of physically performing this -    conveying of source, or (2) access to copy the -    Corresponding Source from a network server at no charge. - -    c) Convey individual copies of the object code with a copy of the -    written offer to provide the Corresponding Source.  This -    alternative is allowed only occasionally and noncommercially, and -    only if you received the object code with such an offer, in accord -    with subsection 6b. - -    d) Convey the object code by offering access from a designated -    place (gratis or for a charge), and offer equivalent access to the -    Corresponding Source in the same way through the same place at no -    further charge.  You need not require recipients to copy the -    Corresponding Source along with the object code.  If the place to -    copy the object code is a network server, the Corresponding Source -    may be on a different server (operated by you or a third party) -    that supports equivalent copying facilities, provided you maintain -    clear directions next to the object code saying where to find the -    Corresponding Source.  Regardless of what server hosts the -    Corresponding Source, you remain obligated to ensure that it is -    available for as long as needed to satisfy these requirements. - -    e) Convey the object code using peer-to-peer transmission, provided -    you inform other peers where the object code and Corresponding -    Source of the work are being offered to the general public at no -    charge under subsection 6d. - -  A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - -  A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling.  In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage.  For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product.  A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - -  "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source.  The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - -  If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information.  But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - -  The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed.  Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - -  Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - -  7. Additional Terms. - -  "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law.  If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - -  When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it.  (Additional permissions may be written to require their own -removal in certain cases when you modify the work.)  You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - -  Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - -    a) Disclaiming warranty or limiting liability differently from the -    terms of sections 15 and 16 of this License; or - -    b) Requiring preservation of specified reasonable legal notices or -    author attributions in that material or in the Appropriate Legal -    Notices displayed by works containing it; or - -    c) Prohibiting misrepresentation of the origin of that material, or -    requiring that modified versions of such material be marked in -    reasonable ways as different from the original version; or - -    d) Limiting the use for publicity purposes of names of licensors or -    authors of the material; or - -    e) Declining to grant rights under trademark law for use of some -    trade names, trademarks, or service marks; or - -    f) Requiring indemnification of licensors and authors of that -    material by anyone who conveys the material (or modified versions of -    it) with contractual assumptions of liability to the recipient, for -    any liability that these contractual assumptions directly impose on -    those licensors and authors. - -  All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10.  If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term.  If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - -  If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - -  Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - -  8. Termination. - -  You may not propagate or modify a covered work except as expressly -provided under this License.  Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - -  However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - -  Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - -  Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License.  If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - -  9. Acceptance Not Required for Having Copies. - -  You are not required to accept this License in order to receive or -run a copy of the Program.  Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance.  However, -nothing other than this License grants you permission to propagate or -modify any covered work.  These actions infringe copyright if you do -not accept this License.  Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - -  10. Automatic Licensing of Downstream Recipients. - -  Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License.  You are not responsible -for enforcing compliance by third parties with this License. - -  An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations.  If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - -  You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License.  For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - -  11. Patents. - -  A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based.  The -work thus licensed is called the contributor's "contributor version". - -  A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version.  For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - -  Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - -  In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement).  To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - -  If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients.  "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - -  If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - -  A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License.  You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - -  Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - -  12. No Surrender of Others' Freedom. - -  If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License.  If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all.  For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - -  13. Use with the GNU Affero General Public License. - -  Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work.  The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - -  14. Revised Versions of this License. - -  The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time.  Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - -  Each version is given a distinguishing version number.  If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation.  If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - -  If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - -  Later license versions may give you additional or different -permissions.  However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - -  15. Disclaimer of Warranty. - -  THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - -  16. Limitation of Liability. - -  IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - -  17. Interpretation of Sections 15 and 16. - -  If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - -                     END OF TERMS AND CONDITIONS - -            How to Apply These Terms to Your New Programs - -  If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - -  To do so, attach the following notices to the program.  It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - -    <one line to give the program's name and a brief idea of what it does.> -    Copyright (C) <year>  <name of author> - -    This program is free software: you can redistribute it and/or modify -    it under the terms of the GNU General Public License as published by -    the Free Software Foundation, either version 3 of the License, or -    (at your option) any later version. - -    This program is distributed in the hope that it will be useful, -    but WITHOUT ANY WARRANTY; without even the implied warranty of -    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the -    GNU General Public License for more details. - -    You should have received a copy of the GNU General Public License -    along with this program.  If not, see <https://www.gnu.org/licenses/>. - -Also add information on how to contact you by electronic and paper mail. - -  If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - -    <program>  Copyright (C) <year>  <name of author> -    This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. -    This is free software, and you are welcome to redistribute it -    under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License.  Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - -  You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -<https://www.gnu.org/licenses/>. - -  The GNU General Public License does not permit incorporating your program -into proprietary programs.  If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library.  If this is what you want to do, use the GNU Lesser General -Public License instead of this License.  But first, please read -<https://www.gnu.org/licenses/why-not-lgpl.html>. +MIT/X Consortium License + +© 2009-2011 Enno Boland <g s01 de> +© 2011 Connor Lane Smith <cls@lubutu.com> +© 2012 Christoph Lohmann <20h@r-36.net>  + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the "Software"), +to deal in the Software without restriction, including without limitation +the rights to use, copy, modify, merge, publish, distribute, sublicense, +and/or sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL +THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..32cc25b --- /dev/null +++ b/Makefile @@ -0,0 +1,60 @@ +# tabbed - tabbing interface +# See LICENSE file for copyright and license details. + +include config.mk + +SRC = tabbed.c +OBJ = ${SRC:.c=.o} + +all: options tabbed + +options: +	@echo tabbed build options: +	@echo "CFLAGS   = ${CFLAGS}" +	@echo "LDFLAGS  = ${LDFLAGS}" +	@echo "CC       = ${CC}" + +.c.o: +	@echo CC $< +	@${CC} -c ${CFLAGS} $< + +${OBJ}: config.h config.mk + +config.h: +	@echo creating $@ from config.def.h +	@cp config.def.h $@ + +tabbed: tabbed.o +	@echo CC -o $@ +	@${CC} -o $@ tabbed.o ${LDFLAGS} + +clean: +	@echo cleaning +	@rm -f tabbed ${OBJ} tabbed-${VERSION}.tar.gz + +dist: clean +	@echo creating dist tarball +	@mkdir -p tabbed-${VERSION} +	@cp -R LICENSE Makefile README config.def.h config.mk \ +		tabbed.1 arg.h ${SRC} tabbed-${VERSION} +	@tar -cf tabbed-${VERSION}.tar tabbed-${VERSION} +	@gzip tabbed-${VERSION}.tar +	@rm -rf tabbed-${VERSION} + +install: all +	@echo installing executable file to ${DESTDIR}${PREFIX}/bin +	@mkdir -p ${DESTDIR}${PREFIX}/bin +	@cp -f tabbed ${DESTDIR}${PREFIX}/bin +	@chmod 755 ${DESTDIR}${PREFIX}/bin/tabbed +	@echo installing manual page to ${DESTDIR}${MANPREFIX}/man1 +	@mkdir -p ${DESTDIR}${MANPREFIX}/man1 +	@sed "s/VERSION/${VERSION}/g" < tabbed.1 > ${DESTDIR}${MANPREFIX}/man1/tabbed.1 +	@chmod 644 ${DESTDIR}${MANPREFIX}/man1/tabbed.1 + +uninstall: +	@echo removing executable file from ${DESTDIR}${PREFIX}/bin +	@rm -f ${DESTDIR}${PREFIX}/bin/tabbed +	@echo removing manual page from ${DESTDIR}${MANPREFIX}/man1 +	@rm -f ${DESTDIR}${MANPREFIX}/man1/tabbed.1 + +.PHONY: all options clean dist install uninstall @@ -0,0 +1,22 @@ +tabbed - generic tabbed interface +================================= +tabbed is a simple tabbed X window container. + +Requirements +------------ +In order to build tabbed you need the Xlib header files. + +Installation +------------ +Edit config.mk to match your local setup (tabbed is installed into +the /usr/local namespace by default). + +Afterwards enter the following command to build and install tabbed +(if necessary as root): + +    make clean install + +Running tabbed +-------------- +See the man page for details. + @@ -0,0 +1,4 @@ +# TODO +* add some way to detach windows +* add some way to attach windows + @@ -0,0 +1,52 @@ +/* See the LICENSE file for copyright and license details. */ + +#ifndef __ARG_H__ +#define __ARG_H__ + +extern char *argv0; + +#define USED(x)		((void)(x)) + +/* use main(int argc, char *argv[]) */ +#define ARGBEGIN	for (argv0 = *argv, argv++, argc--;\ +					argv[0] && argv[0][1]\ +					&& argv[0][0] == '-';\ +					argc--, argv++) {\ +				char _argc;\ +				char **_argv;\ +				int brk;\ +				if (argv[0][1] == '-' && argv[0][2] == '\0') {\ +					argv++;\ +					argc--;\ +					break;\ +				}\ +				for (brk = 0, argv[0]++, _argv = argv;\ +						argv[0][0] && !brk;\ +						argv[0]++) {\ +					if (_argv != argv)\ +						break;\ +					_argc = argv[0][0];\ +					switch (_argc) + +#define ARGEND			}\ +				USED(_argc);\ +			}\ +			USED(argv);\ +			USED(argc); + +#define ARGC()		_argc + +#define EARGF(x)	((argv[0][1] == '\0' && argv[1] == NULL)?\ +				((x), abort(), (char *)0) :\ +				(brk = 1, (argv[0][1] != '\0')?\ +					(&argv[0][1]) :\ +					(argc--, argv++, argv[0]))) + +#define ARGF()		((argv[0][1] == '\0' && argv[1] == NULL)?\ +				(char *)0 :\ +				(brk = 1, (argv[0][1] != '\0')?\ +					(&argv[0][1]) :\ +					(argc--, argv++, argv[0]))) + +#endif + diff --git a/config.def.h b/config.def.h new file mode 100644 index 0000000..b45bea6 --- /dev/null +++ b/config.def.h @@ -0,0 +1,74 @@ +/* See LICENSE file for copyright and license details. */ + +/* appearance */ +static char font[] = +    "DejaVuSansMono Nerd Font Mono:pixelsize=14;antialias=true;autohint=true"; +static char *normbgcolor = "#222222"; +static char *normfgcolor = "#cccccc"; +static char *selbgcolor = "#555555"; +static char *selfgcolor = "#ffffff"; +static char *urgbgcolor = "#111111"; +static char *urgfgcolor = "#cc0000"; +static const char before[] = "<"; +static const char after[] = ">"; +static const int tabwidth = 200; +static const Bool foreground = True; + +/* + * Where to place a new tab when it is opened. When npisrelative is True, + * then the current position is changed + newposition. If npisrelative + * is False, then newposition is an absolute position. + */ +static int newposition = 0; +static Bool npisrelative = False; + +#define SETPROP(p)                                                             \ +  {                                                                            \ +    .v = (char *[]) {                                                          \ +      "/bin/sh", "-c",                                                         \ +          "prop=\"`xwininfo -children -id $1 | grep '^     0x' | sed -e's@^ "  \ +          "*\\(0x[0-9a-f]*\\) \"\\([^\"]*\\)\".*@\\1 \\2@' | xargs -0 printf " \ +          "%b | dmenu -l 10`\" &&"                                             \ +          "xprop -id $1 -f $0 8s -set $0 \"$prop\"",                           \ +          p, winid, NULL                                                       \ +    }                                                                          \ +  } + +/* + * Xresources preferences to load at startup + */ +ResourcePref resources[] = { +    {"font", STRING, &font},          {"color0", STRING, &normbgcolor}, +    {"color4", STRING, &normfgcolor}, {"color4", STRING, &selbgcolor}, +    {"color7", STRING, &selfgcolor},  {"color2", STRING, &urgbgcolor}, +    {"color3", STRING, &urgfgcolor}, +}; + +#define MODKEY ControlMask +static Key keys[] = { +    /* modifier                     key        function        argument */ +    {MODKEY | ShiftMask, XK_Return, focusonce, {0}}, +    {MODKEY | ShiftMask, XK_Return, spawn, {0}}, +    {MODKEY, XK_t, spawn, SETPROP("_TABBED_SELECT_TAB")}, + +    {MODKEY | ShiftMask, XK_l, rotate, {.i = +1}}, +    {MODKEY | ShiftMask, XK_h, rotate, {.i = -1}}, +    {MODKEY | ShiftMask, XK_j, movetab, {.i = -1}}, +    {MODKEY | ShiftMask, XK_k, movetab, {.i = +1}}, +    {MODKEY, XK_Tab, rotate, {.i = 0}}, + +    {MODKEY, XK_1, move, {.i = 0}}, +    {MODKEY, XK_2, move, {.i = 1}}, +    {MODKEY, XK_3, move, {.i = 2}}, +    {MODKEY, XK_4, move, {.i = 3}}, +    {MODKEY, XK_5, move, {.i = 4}}, +    {MODKEY, XK_6, move, {.i = 5}}, +    {MODKEY, XK_7, move, {.i = 6}}, +    {MODKEY, XK_8, move, {.i = 7}}, +    {MODKEY, XK_9, move, {.i = 8}}, +    {MODKEY, XK_0, move, {.i = 9}}, + +    {MODKEY, XK_q, killclient, {0}}, + +    {0, XK_F11, fullscreen, {0}}, +}; diff --git a/config.h b/config.h new file mode 100644 index 0000000..b45bea6 --- /dev/null +++ b/config.h @@ -0,0 +1,74 @@ +/* See LICENSE file for copyright and license details. */ + +/* appearance */ +static char font[] = +    "DejaVuSansMono Nerd Font Mono:pixelsize=14;antialias=true;autohint=true"; +static char *normbgcolor = "#222222"; +static char *normfgcolor = "#cccccc"; +static char *selbgcolor = "#555555"; +static char *selfgcolor = "#ffffff"; +static char *urgbgcolor = "#111111"; +static char *urgfgcolor = "#cc0000"; +static const char before[] = "<"; +static const char after[] = ">"; +static const int tabwidth = 200; +static const Bool foreground = True; + +/* + * Where to place a new tab when it is opened. When npisrelative is True, + * then the current position is changed + newposition. If npisrelative + * is False, then newposition is an absolute position. + */ +static int newposition = 0; +static Bool npisrelative = False; + +#define SETPROP(p)                                                             \ +  {                                                                            \ +    .v = (char *[]) {                                                          \ +      "/bin/sh", "-c",                                                         \ +          "prop=\"`xwininfo -children -id $1 | grep '^     0x' | sed -e's@^ "  \ +          "*\\(0x[0-9a-f]*\\) \"\\([^\"]*\\)\".*@\\1 \\2@' | xargs -0 printf " \ +          "%b | dmenu -l 10`\" &&"                                             \ +          "xprop -id $1 -f $0 8s -set $0 \"$prop\"",                           \ +          p, winid, NULL                                                       \ +    }                                                                          \ +  } + +/* + * Xresources preferences to load at startup + */ +ResourcePref resources[] = { +    {"font", STRING, &font},          {"color0", STRING, &normbgcolor}, +    {"color4", STRING, &normfgcolor}, {"color4", STRING, &selbgcolor}, +    {"color7", STRING, &selfgcolor},  {"color2", STRING, &urgbgcolor}, +    {"color3", STRING, &urgfgcolor}, +}; + +#define MODKEY ControlMask +static Key keys[] = { +    /* modifier                     key        function        argument */ +    {MODKEY | ShiftMask, XK_Return, focusonce, {0}}, +    {MODKEY | ShiftMask, XK_Return, spawn, {0}}, +    {MODKEY, XK_t, spawn, SETPROP("_TABBED_SELECT_TAB")}, + +    {MODKEY | ShiftMask, XK_l, rotate, {.i = +1}}, +    {MODKEY | ShiftMask, XK_h, rotate, {.i = -1}}, +    {MODKEY | ShiftMask, XK_j, movetab, {.i = -1}}, +    {MODKEY | ShiftMask, XK_k, movetab, {.i = +1}}, +    {MODKEY, XK_Tab, rotate, {.i = 0}}, + +    {MODKEY, XK_1, move, {.i = 0}}, +    {MODKEY, XK_2, move, {.i = 1}}, +    {MODKEY, XK_3, move, {.i = 2}}, +    {MODKEY, XK_4, move, {.i = 3}}, +    {MODKEY, XK_5, move, {.i = 4}}, +    {MODKEY, XK_6, move, {.i = 5}}, +    {MODKEY, XK_7, move, {.i = 6}}, +    {MODKEY, XK_8, move, {.i = 7}}, +    {MODKEY, XK_9, move, {.i = 8}}, +    {MODKEY, XK_0, move, {.i = 9}}, + +    {MODKEY, XK_q, killclient, {0}}, + +    {0, XK_F11, fullscreen, {0}}, +}; diff --git a/config.mk b/config.mk new file mode 100644 index 0000000..037f9d7 --- /dev/null +++ b/config.mk @@ -0,0 +1,25 @@ +# tabbed version +VERSION = 0.6 + +# Customize below to fit your system + +# paths +PREFIX = /usr/local +MANPREFIX = ${PREFIX}/share/man + +# includes and libs +INCS = -I. -I/usr/include -I/usr/include/freetype2 +LIBS = -L/usr/lib -lc -lX11 -lfontconfig -lXft + +# flags +CPPFLAGS = -DVERSION=\"${VERSION}\" -D_BSD_SOURCE +CFLAGS = -std=c99 -pedantic -Wall -O0 ${INCS} ${CPPFLAGS} +LDFLAGS = -s ${LIBS} + +# Solaris +#CFLAGS = -fast ${INCS} -DVERSION=\"${VERSION}\" +#LDFLAGS = ${LIBS} + +# compiler and linker +CC = cc + diff --git a/tabbed-0.6-xft.diff b/tabbed-0.6-xft.diff new file mode 100644 index 0000000..a6ed881 --- /dev/null +++ b/tabbed-0.6-xft.diff @@ -0,0 +1,19 @@ +@@ -1040,15 +1000,9 @@ spawn(const Arg *arg) { +  + int + textnw(const char *text, unsigned int len) { +-	XRectangle r; +- +-	if(dc.font.set) { +-		XmbTextExtents(dc.font.set, text, len, NULL, &r); +- +-		return r.width; +-	} +- +-	return XTextWidth(dc.font.xfont, text, len); ++	XGlyphInfo ext; ++	XftTextExtentsUtf8(dpy, dc.font.xfont, (XftChar8 *) text, len, &ext); ++	return ext.xOff; + } +  + void diff --git a/tabbed-clientnumber-0.6.diff b/tabbed-clientnumber-0.6.diff new file mode 100644 index 0000000..430245c --- /dev/null +++ b/tabbed-clientnumber-0.6.diff @@ -0,0 +1,23 @@ +diff --git a/tabbed.c b/tabbed.c +index d30206b..70642cb 100644 +--- a/tabbed.c ++++ b/tabbed.c +@@ -308,6 +308,7 @@ drawbar(void) { + 	unsigned long *col; + 	int c, fc, width, n = 0; + 	char *name = NULL; ++	char tabtitle[256]; +  + 	if(nclients == 0) { + 		dc.x = 0; +@@ -353,7 +354,9 @@ drawbar(void) { + 		} else { + 			col = dc.norm; + 		} +-		drawtext(clients[c]->name, col); ++		snprintf(tabtitle, sizeof(tabtitle), "%d: %s", ++		         c + 1, clients[c]->name); ++		drawtext(tabtitle, col); + 		dc.x += dc.w; + 		clients[c]->tabx = dc.x; + 	} diff --git a/tabbed-xresources-20210317-dabf6a2.diff b/tabbed-xresources-20210317-dabf6a2.diff new file mode 100644 index 0000000..e7f72fa --- /dev/null +++ b/tabbed-xresources-20210317-dabf6a2.diff @@ -0,0 +1,15 @@ +diff --git a/tabbed.c b/tabbed.c +index eafe28a..c5bffc7 100644 +--- a/tabbed.c ++++ b/tabbed.c +@@ -1354,6 +1422,7 @@ main(int argc, char *argv[]) + 	if (!(dpy = XOpenDisplay(NULL))) + 		die("%s: cannot open display\n", argv0); +  ++	config_init(); + 	setup(); + 	printf("0x%lx\n", win); + 	fflush(NULL); +--  +2.30.2 + diff --git a/tabbed.1 b/tabbed.1 new file mode 100644 index 0000000..0ae29ce --- /dev/null +++ b/tabbed.1 @@ -0,0 +1,144 @@ +.TH TABBED 1 tabbed\-VERSION +.SH NAME +tabbed \- generic tabbed interface +.SH SYNOPSIS +.B tabbed +.RB [ \-c ] +.RB [ \-d ] +.RB [ \-h ] +.RB [ \-s ] +.RB [ \-v ] +.RB [ \-g +.IR geometry ] +.RB [ \-n +.IR name ] +.RB [ \-p +.IR [ s +/- ] pos ] +.RB [ \-r +.IR narg ] +.IR [ command ... ] +.SH DESCRIPTION +.B tabbed +is a simple tabbed container for applications which support XEmbed. Tabbed +will then run the provided command with the xid of tabbed as appended +argument. (See EXAMPLES.) The automatic spawning of the command can be +disabled by providing the -s parameter. If no command is provided +tabbed will just print its xid and run no command. +.SH OPTIONS +.TP +.B \-c +close tabbed when the last tab is closed. Mutually exclusive with -f. +.TP +.B \-d +detaches tabbed from the terminal and prints its XID to stdout. +.TP +.B \-f +fill up tabbed again by spawning the provided command, when the last tab is +closed. Mutually exclusive with -c. +.TP +.B \-h +will print the usage of tabbed. +.TP +.BI \-g " geometry" +defines the X11 geometry string, which will fixate the height and width of +tabbed. +Them form is [=][<width>{xX}<height>][{+-}<xoffset>{+-}<yoffset>]. See +.BR XParseGeometry (3) +for further details. +.TP +.BI \-n " name" +will set the WM_CLASS attribute to +.I name. +.TP +.BI \-p " [ s +/-] pos" +will set the absolute or relative position of where to start a new tab. When +.I pos +is is given without 's' in front it is an absolute position. Then negative +numbers will be the position from the last tab, where -1 is the last tab. +If 's' is given, then +.I pos +is a relative position to the current selected tab. If this reaches the limits +of the tabs; those limits then apply. +.TP +.BI \-r " narg" +will replace the +.I narg +th argument in +.I command +with the window id, rather than appending it to the end. +.TP +.B \-s +will disable automatic spawning of the command. +.TP +.BI \-t " color" +defines the selected background color. +.IR #RGB , +.IR #RRGGBB , +and X color names are supported. +.TP +.BI \-T " color" +defines the selected foreground color. +.TP +.BI \-u " color" +defines the normal background color. +.TP +.BI \-U " color" +defines the normal foreground color. +.TP +.B \-v +prints version information to stderr, then exits. +.SH USAGE +.TP +.B Ctrl\-Shift\-Return +open new tab +.TP +.B Ctrl\-Shift\-h +previous tab +.TP +.B Ctrl\-Shift\-l +next tab +.TP +.B Ctrl\-Shift\-j +move selected tab one to the left +.TP +.B Ctrl\-Shift\-k +move selected tab one to the right +.TP +.B Ctrl\-Tab +toggle between the selected and last selected tab +.TP +.B Ctrl\-t +open dmenu to either create a new tab appending the entered string or select +an already existing tab. +.TP +.B Ctrl\-q +close tab +.TP +.B Ctrl\-[0..9] +jumps to nth tab +.TP +.B F11 +Toggle fullscreen mode. +.SH EXAMPLES +$ tabbed surf -e +.TP +$ tabbed urxvt -embed +.TP +$ tabbed xterm -into +.TP +$ $(tabbed -d >/tmp/tabbed.xid); urxvt -embed $(</tmp/tabbed.xid); +.TP +$ tabbed -r 2 st -w '' -e tmux +.SH CUSTOMIZATION +.B tabbed +can be customized by creating a custom config.h and (re)compiling the source +code. This keeps it fast, secure and simple. +.SH AUTHORS +See the LICENSE file for the authors. +.SH LICENSE +See the LICENSE file for the terms of redistribution. +.SH SEE ALSO +.BR st (1) +.SH BUGS +Please report them. + diff --git a/tabbed.c b/tabbed.c new file mode 100644 index 0000000..097c9ca --- /dev/null +++ b/tabbed.c @@ -0,0 +1,1261 @@ +/* + * See LICENSE file for copyright and license details. + */ + +#include <X11/XKBlib.h> +#include <X11/Xatom.h> +#include <X11/Xft/Xft.h> +#include <X11/Xlib.h> +#include <X11/Xproto.h> +#include <X11/Xresource.h> +#include <X11/Xutil.h> +#include <locale.h> +#include <signal.h> +#include <stdarg.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/wait.h> +#include <unistd.h> + +#include "arg.h" + +/* XEMBED messages */ +#define XEMBED_EMBEDDED_NOTIFY 0 +#define XEMBED_WINDOW_ACTIVATE 1 +#define XEMBED_WINDOW_DEACTIVATE 2 +#define XEMBED_REQUEST_FOCUS 3 +#define XEMBED_FOCUS_IN 4 +#define XEMBED_FOCUS_OUT 5 +#define XEMBED_FOCUS_NEXT 6 +#define XEMBED_FOCUS_PREV 7 +/* 8-9 were used for XEMBED_GRAB_KEY/XEMBED_UNGRAB_KEY */ +#define XEMBED_MODALITY_ON 10 +#define XEMBED_MODALITY_OFF 11 +#define XEMBED_REGISTER_ACCELERATOR 12 +#define XEMBED_UNREGISTER_ACCELERATOR 13 +#define XEMBED_ACTIVATE_ACCELERATOR 14 + +/* Details for  XEMBED_FOCUS_IN: */ +#define XEMBED_FOCUS_CURRENT 0 +#define XEMBED_FOCUS_FIRST 1 +#define XEMBED_FOCUS_LAST 2 + +/* Macros */ +#define MAX(a, b) ((a) > (b) ? (a) : (b)) +#define MIN(a, b) ((a) < (b) ? (a) : (b)) +#define LENGTH(x) (sizeof((x)) / sizeof(*(x))) +#define CLEANMASK(mask) (mask & ~(numlockmask | LockMask)) +#define TEXTW(x) (textnw(x, strlen(x)) + dc.font.height) + +enum { ColFG, ColBG, ColLast }; /* color */ +enum { +  WMProtocols, +  WMDelete, +  WMName, +  WMState, +  WMFullscreen, +  XEmbed, +  WMSelectTab, +  WMLast +}; /* default atoms */ + +typedef union { +  int i; +  const void *v; +} Arg; + +typedef struct { +  unsigned int mod; +  KeySym keysym; +  void (*func)(const Arg *); +  const Arg arg; +} Key; + +typedef struct { +  int x, y, w, h; +  XftColor norm[ColLast]; +  XftColor sel[ColLast]; +  Drawable drawable; +  GC gc; +  struct { +    int ascent; +    int descent; +    int height; +    XftFont *xfont; +  } font; +} DC; /* draw context */ + +typedef struct Client { +  char name[256]; +  Window win; +  int tabx; +  Bool mapped; +  Bool closed; +} Client; + +/* Xresources preferences */ +enum resource_type { STRING = 0, INTEGER = 1, FLOAT = 2 }; + +typedef struct { +  char *name; +  enum resource_type type; +  void *dst; +} ResourcePref; + +/* function declarations */ +static void buttonpress(const XEvent *e); +static void cleanup(void); +static void clientmessage(const XEvent *e); +static void config_init(void); +static void configurenotify(const XEvent *e); +static void configurerequest(const XEvent *e); +static void createnotify(const XEvent *e); +static void destroynotify(const XEvent *e); +static void die(const char *errstr, ...); +static void drawbar(void); +static void drawtext(const char *text, XftColor col[ColLast]); +static void *emallocz(size_t size); +static void *erealloc(void *o, size_t size); +static void expose(const XEvent *e); +static void focus(int c); +static void focusin(const XEvent *e); +static void focusonce(const Arg *arg); +static void fullscreen(const Arg *arg); +static char *getatom(int a); +static int getclient(Window w); +static XftColor getcolor(const char *colstr); +static int getfirsttab(void); +static Bool gettextprop(Window w, Atom atom, char *text, unsigned int size); +static void initfont(const char *fontstr); +static Bool isprotodel(int c); +static void keypress(const XEvent *e); +static void killclient(const Arg *arg); +static void manage(Window win); +static void maprequest(const XEvent *e); +static void move(const Arg *arg); +static void movetab(const Arg *arg); +static void propertynotify(const XEvent *e); +static void resize(int c, int w, int h); +static int resource_load(XrmDatabase db, char *name, enum resource_type rtype, +                         void *dst); +static void rotate(const Arg *arg); +static void run(void); +static void sendxembed(int c, long msg, long detail, long d1, long d2); +static void setup(void); +static void setcmd(int argc, char *argv[], int); +static void sigchld(int unused); +static void spawn(const Arg *arg); +static int textnw(const char *text, unsigned int len); +static void unmanage(int c); +static void updatenumlockmask(void); +static void updatetitle(int c); +static int xerror(Display *dpy, XErrorEvent *ee); +static void xsettitle(Window w, const char *str); + +/* variables */ +static int screen; +static void (*handler[LASTEvent])(const XEvent *) = { +    [ButtonPress] = buttonpress, +    [ClientMessage] = clientmessage, +    [ConfigureNotify] = configurenotify, +    [ConfigureRequest] = configurerequest, +    [CreateNotify] = createnotify, +    [DestroyNotify] = destroynotify, +    [Expose] = expose, +    [FocusIn] = focusin, +    [KeyPress] = keypress, +    [MapRequest] = maprequest, +    [PropertyNotify] = propertynotify, +}; +static int bh, wx, wy, ww, wh; +static unsigned int numlockmask = 0; +static Bool running = True, nextfocus, doinitspawn = True, fillagain = False, +            closelastclient = False; +static Display *dpy; +static DC dc; +static Atom wmatom[WMLast]; +static Window root, win; +static Client **clients = NULL; +static int nclients = 0, sel = -1, lastsel = -1; +static int (*xerrorxlib)(Display *, XErrorEvent *); +static int cmd_append_pos = 0; +static char winid[64]; +static char **cmd = NULL; +static char *wmname = "tabbed"; +static const char *geometry = NULL; + +char *argv0; + +/* configuration, allows nested code to access above variables */ +#include "config.h" + +void buttonpress(const XEvent *e) { +  const XButtonPressedEvent *ev = &e->xbutton; +  int i; +  int fc; +  Arg arg; + +  fc = getfirsttab(); + +  if ((fc > 0 && ev->x < TEXTW(before)) || ev->x < 0) +    return; + +  if (ev->y < 0 || ev->y > bh) +    return; + +  for (i = (fc > 0) ? fc : 0; i < nclients; i++) { +    if (clients[i]->tabx > ev->x) { +      switch (ev->button) { +      case Button1: +        focus(i); +        break; +      case Button2: +        focus(i); +        killclient(NULL); +        break; +      case Button4: +      case Button5: +        arg.i = ev->button == Button4 ? -1 : 1; +        rotate(&arg); +        break; +      } +      break; +    } +  } +} + +void cleanup(void) { +  int i; + +  for (i = 0; i < nclients; i++) { +    focus(i); +    killclient(NULL); +    killclient(NULL); +    XReparentWindow(dpy, clients[i]->win, root, 0, 0); +    unmanage(i); +  } +  free(clients); +  clients = NULL; + +  XFreePixmap(dpy, dc.drawable); +  XFreeGC(dpy, dc.gc); +  XDestroyWindow(dpy, win); +  XSync(dpy, False); +  free(cmd); +} + +void clientmessage(const XEvent *e) { +  const XClientMessageEvent *ev = &e->xclient; + +  if (ev->message_type == wmatom[WMProtocols] && +      ev->data.l[0] == wmatom[WMDelete]) { +    running = False; +  } +} + +void config_init(void) { +  char *resm; +  XrmDatabase db; +  ResourcePref *p; + +  XrmInitialize(); +  resm = XResourceManagerString(dpy); +  if (!resm) +    return; + +  db = XrmGetStringDatabase(resm); +  for (p = resources; p < resources + LENGTH(resources); p++) +    resource_load(db, p->name, p->type, p->dst); +} + +void configurenotify(const XEvent *e) { +  const XConfigureEvent *ev = &e->xconfigure; + +  if (ev->window == win && (ev->width != ww || ev->height != wh)) { +    ww = ev->width; +    wh = ev->height; +    XFreePixmap(dpy, dc.drawable); +    dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen)); +    if (sel > -1) +      resize(sel, ww, wh - bh); +    XSync(dpy, False); +  } +} + +void configurerequest(const XEvent *e) { +  const XConfigureRequestEvent *ev = &e->xconfigurerequest; +  XWindowChanges wc; +  int c; + +  if ((c = getclient(ev->window)) > -1) { +    wc.x = 0; +    wc.y = bh; +    wc.width = ww; +    wc.height = wh - bh; +    wc.border_width = 0; +    wc.sibling = ev->above; +    wc.stack_mode = ev->detail; +    XConfigureWindow(dpy, clients[c]->win, ev->value_mask, &wc); +  } +} + +void createnotify(const XEvent *e) { +  const XCreateWindowEvent *ev = &e->xcreatewindow; + +  if (ev->window != win && getclient(ev->window) < 0) +    manage(ev->window); +} + +void destroynotify(const XEvent *e) { +  const XDestroyWindowEvent *ev = &e->xdestroywindow; +  int c; + +  if ((c = getclient(ev->window)) > -1) +    unmanage(c); +} + +void die(const char *errstr, ...) { +  va_list ap; + +  va_start(ap, errstr); +  vfprintf(stderr, errstr, ap); +  va_end(ap); +  exit(EXIT_FAILURE); +} + +void drawbar(void) { +  XftColor *col; +  int c, fc, width, n = 0; +  char *name = NULL; +  char tabtitle[256]; + +  if (nclients == 0) { +    dc.x = 0; +    dc.w = ww; +    XFetchName(dpy, win, &name); +    drawtext(name ? name : "", dc.norm); +    XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, bh, 0, 0); +    XSync(dpy, False); + +    return; +  } + +  width = ww; +  clients[nclients - 1]->tabx = -1; +  fc = getfirsttab(); +  if (fc > -1) +    n = nclients - fc; + +  if ((n * tabwidth) > width) { +    dc.w = TEXTW(after); +    dc.x = width - dc.w; +    drawtext(after, dc.sel); +    width -= dc.w; +  } +  dc.x = 0; + +  if (fc > 0) { +    dc.w = TEXTW(before); +    drawtext(before, dc.sel); +    dc.x += dc.w; +    width -= dc.w; +  } + +  for (c = (fc > 0) ? fc : 0; c < nclients && dc.x < width; c++) { +    dc.w = tabwidth; +    if (c == sel) { +      col = dc.sel; +      if ((n * tabwidth) > width) { +        dc.w += width % tabwidth; +      } else { +        dc.w = width - (n - 1) * tabwidth; +      } +    } else { +      col = dc.norm; +    } +    snprintf(tabtitle, sizeof(tabtitle), "%d: %s", c + 1, clients[c]->name); +    drawtext(tabtitle, col); +    dc.x += dc.w; +    clients[c]->tabx = dc.x; +  } +  XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, bh, 0, 0); +  XSync(dpy, False); +} + +void drawtext(const char *text, XftColor col[ColLast]) { +  int i, x, y, h, len, olen; +  char buf[256]; +  XftDraw *d; +  XRectangle r = {dc.x, dc.y, dc.w, dc.h}; + +  XSetForeground(dpy, dc.gc, col[ColBG].pixel); +  XFillRectangles(dpy, dc.drawable, dc.gc, &r, 1); +  if (!text) +    return; + +  olen = strlen(text); +  h = dc.font.ascent + dc.font.descent; +  y = dc.y + (dc.h / 2) - (h / 2) + dc.font.ascent; +  x = dc.x + (h / 2); + +  /* shorten text if necessary */ +  for (len = MIN(olen, sizeof(buf)); len && textnw(text, len) > dc.w - h; len--) +    ; +  if (!len) +    return; + +  memcpy(buf, text, len); +  if (len < olen) { +    for (i = len; i && i > len - 3; buf[--i] = '.') +      ; +  } +  d = XftDrawCreate(dpy, dc.drawable, DefaultVisual(dpy, screen), +                    DefaultColormap(dpy, screen)); + +  XftDrawStringUtf8(d, &col[ColFG], dc.font.xfont, x, y, (XftChar8 *)buf, len); +  XftDrawDestroy(d); +} + +void *emallocz(size_t size) { +  void *p; + +  if (!(p = calloc(1, size))) +    die("tabbed: cannot malloc\n"); +  return p; +} + +void *erealloc(void *o, size_t size) { +  void *p; + +  if (!(p = realloc(o, size))) +    die("tabbed: cannot realloc\n"); +  return p; +} + +void expose(const XEvent *e) { +  const XExposeEvent *ev = &e->xexpose; + +  if (ev->count == 0 && win == ev->window) +    drawbar(); +} + +void focus(int c) { +  char buf[BUFSIZ] = "tabbed-" VERSION " ::"; +  size_t i, n; + +  /* If c, sel and clients are -1, raise tabbed-win itself */ +  if (nclients == 0) { +    cmd[cmd_append_pos] = NULL; +    for (i = 0, n = strlen(buf); cmd[i] && n < sizeof(buf); i++) +      n += snprintf(&buf[n], sizeof(buf) - n, " %s", cmd[i]); + +    xsettitle(win, buf); +    XRaiseWindow(dpy, win); + +    return; +  } + +  if (c < 0 || c >= nclients) +    return; + +  resize(c, ww, wh - bh); +  XRaiseWindow(dpy, clients[c]->win); +  XSetInputFocus(dpy, clients[c]->win, RevertToParent, CurrentTime); +  sendxembed(c, XEMBED_FOCUS_IN, XEMBED_FOCUS_CURRENT, 0, 0); +  sendxembed(c, XEMBED_WINDOW_ACTIVATE, 0, 0, 0); +  xsettitle(win, clients[c]->name); + +  /* If sel is already c, change nothing. */ +  if (sel != c) { +    lastsel = sel; +    sel = c; +  } + +  drawbar(); +  XSync(dpy, False); +} + +void focusin(const XEvent *e) { +  const XFocusChangeEvent *ev = &e->xfocus; +  int dummy; +  Window focused; + +  if (ev->mode != NotifyUngrab) { +    XGetInputFocus(dpy, &focused, &dummy); +    if (focused == win) +      focus(sel); +  } +} + +void focusonce(const Arg *arg) { nextfocus = True; } + +void fullscreen(const Arg *arg) { +  XEvent e; + +  e.type = ClientMessage; +  e.xclient.window = win; +  e.xclient.message_type = wmatom[WMState]; +  e.xclient.format = 32; +  e.xclient.data.l[0] = 2; +  e.xclient.data.l[1] = wmatom[WMFullscreen]; +  e.xclient.data.l[2] = 0; +  XSendEvent(dpy, root, False, SubstructureNotifyMask, &e); +} + +char *getatom(int a) { +  static char buf[BUFSIZ]; +  Atom adummy; +  int idummy; +  unsigned long ldummy; +  unsigned char *p = NULL; + +  XGetWindowProperty(dpy, win, wmatom[a], 0L, BUFSIZ, False, XA_STRING, &adummy, +                     &idummy, &ldummy, &ldummy, &p); +  if (p) { +    strncpy(buf, (char *)p, LENGTH(buf) - 1); +  } else { +    buf[0] = '\0'; +  } +  XFree(p); + +  return buf; +} + +int getclient(Window w) { +  int i; + +  for (i = 0; i < nclients; i++) { +    if (clients[i]->win == w) +      return i; +  } + +  return -1; +} + +XftColor getcolor(const char *colstr) { +  XftColor color; + +  if (!XftColorAllocName(dpy, DefaultVisual(dpy, screen), +                         DefaultColormap(dpy, screen), colstr, &color)) +    die("tabbed: cannot allocate color '%s'\n", colstr); + +  return color; +} + +int getfirsttab(void) { +  int c, n, fc; + +  if (sel < 0) +    return -1; + +  c = sel; +  fc = 0; +  n = nclients; +  if ((n * tabwidth) > ww) { +    for (; (c * tabwidth) > (ww / 2) && (n * tabwidth) > ww; c--, n--, fc++) +      ; +  } + +  return fc; +} + +Bool gettextprop(Window w, Atom atom, char *text, unsigned int size) { +  char **list = NULL; +  int n; +  XTextProperty name; + +  if (!text || size == 0) +    return False; + +  text[0] = '\0'; +  XGetTextProperty(dpy, w, &name, atom); +  if (!name.nitems) +    return False; + +  if (name.encoding == XA_STRING) { +    strncpy(text, (char *)name.value, size - 1); +  } else { +    if (XmbTextPropertyToTextList(dpy, &name, &list, &n) >= Success && n > 0 && +        *list) { +      strncpy(text, *list, size - 1); +      XFreeStringList(list); +    } +  } +  text[size - 1] = '\0'; +  XFree(name.value); + +  return True; +} + +void initfont(const char *fontstr) { +  if (!(dc.font.xfont = XftFontOpenName(dpy, screen, fontstr)) && +      !(dc.font.xfont = XftFontOpenName(dpy, screen, "fixed"))) +    die("error, cannot load font: '%s'\n", fontstr); + +  dc.font.ascent = dc.font.xfont->ascent; +  dc.font.descent = dc.font.xfont->descent; +  dc.font.height = dc.font.ascent + dc.font.descent; +} + +Bool isprotodel(int c) { +  int i, n; +  Atom *protocols; +  Bool ret = False; + +  if (XGetWMProtocols(dpy, clients[c]->win, &protocols, &n)) { +    for (i = 0; !ret && i < n; i++) { +      if (protocols[i] == wmatom[WMDelete]) +        ret = True; +    } +    XFree(protocols); +  } + +  return ret; +} + +void keypress(const XEvent *e) { +  const XKeyEvent *ev = &e->xkey; +  unsigned int i; +  KeySym keysym; + +  keysym = XkbKeycodeToKeysym(dpy, (KeyCode)ev->keycode, 0, 0); +  for (i = 0; i < LENGTH(keys); i++) { +    if (keysym == keys[i].keysym && +        CLEANMASK(keys[i].mod) == CLEANMASK(ev->state) && keys[i].func) { +      keys[i].func(&(keys[i].arg)); +    } +  } +} + +void killclient(const Arg *arg) { +  XEvent ev; + +  if (sel < 0) +    return; + +  if (isprotodel(sel) && !clients[sel]->closed) { +    ev.type = ClientMessage; +    ev.xclient.window = clients[sel]->win; +    ev.xclient.message_type = wmatom[WMProtocols]; +    ev.xclient.format = 32; +    ev.xclient.data.l[0] = wmatom[WMDelete]; +    ev.xclient.data.l[1] = CurrentTime; +    XSendEvent(dpy, clients[sel]->win, False, NoEventMask, &ev); +    clients[sel]->closed = True; +  } else { +    XKillClient(dpy, clients[sel]->win); +  } +} + +void manage(Window w) { +  updatenumlockmask(); +  { +    int i, j, nextpos; +    unsigned int modifiers[] = {0, LockMask, numlockmask, +                                numlockmask | LockMask}; +    KeyCode code; +    Client *c; +    XEvent e; + +    XWithdrawWindow(dpy, w, 0); +    XReparentWindow(dpy, w, win, 0, bh); +    XSelectInput(dpy, w, +                 PropertyChangeMask | StructureNotifyMask | EnterWindowMask); +    XSync(dpy, False); + +    for (i = 0; i < LENGTH(keys); i++) { +      if ((code = XKeysymToKeycode(dpy, keys[i].keysym))) { +        for (j = 0; j < LENGTH(modifiers); j++) { +          XGrabKey(dpy, code, keys[i].mod | modifiers[j], w, True, +                   GrabModeAsync, GrabModeAsync); +        } +      } +    } + +    c = emallocz(sizeof(*c)); +    c->win = w; + +    nclients++; +    clients = erealloc(clients, sizeof(Client *) * nclients); + +    if (npisrelative) { +      nextpos = sel + newposition; +    } else { +      if (newposition < 0) { +        nextpos = nclients - newposition; +      } else { +        nextpos = newposition; +      } +    } +    if (nextpos >= nclients) +      nextpos = nclients - 1; +    if (nextpos < 0) +      nextpos = 0; + +    if (nclients > 1 && nextpos < nclients - 1) { +      memmove(&clients[nextpos + 1], &clients[nextpos], +              sizeof(Client *) * (nclients - nextpos - 1)); +    } +    clients[nextpos] = c; +    updatetitle(nextpos); + +    XLowerWindow(dpy, w); +    XMapWindow(dpy, w); + +    e.xclient.window = w; +    e.xclient.type = ClientMessage; +    e.xclient.message_type = wmatom[XEmbed]; +    e.xclient.format = 32; +    e.xclient.data.l[0] = CurrentTime; +    e.xclient.data.l[1] = XEMBED_EMBEDDED_NOTIFY; +    e.xclient.data.l[2] = 0; +    e.xclient.data.l[3] = win; +    e.xclient.data.l[4] = 0; +    XSendEvent(dpy, root, False, NoEventMask, &e); + +    XSync(dpy, False); + +    /* Adjust sel before focus does set it to lastsel. */ +    if (sel >= nextpos) +      sel++; +    focus((nextfocus) ? nextpos : ((sel < 0) ? 0 : sel)); +    nextfocus = foreground; +  } +} + +void maprequest(const XEvent *e) { +  const XMapRequestEvent *ev = &e->xmaprequest; + +  if (getclient(ev->window) < 0) +    manage(ev->window); +} + +void move(const Arg *arg) { +  if (arg->i >= 0 && arg->i < nclients) +    focus(arg->i); +} + +void movetab(const Arg *arg) { +  int c; +  Client *new; + +  if (sel < 0 || (arg->i == 0)) +    return; + +  c = sel + arg->i; +  while (c >= nclients) +    c -= nclients; +  while (c < 0) +    c += nclients; + +  new = clients[c]; +  clients[c] = clients[sel]; +  clients[sel] = new; + +  sel = c; + +  drawbar(); +} + +void propertynotify(const XEvent *e) { +  const XPropertyEvent *ev = &e->xproperty; +  int c; +  char *selection = NULL; +  Arg arg; + +  if (ev->state == PropertyNewValue && ev->atom == wmatom[WMSelectTab]) { +    selection = getatom(WMSelectTab); +    if (!strncmp(selection, "0x", 2)) { +      arg.i = getclient(strtoul(selection, NULL, 0)); +      move(&arg); +    } else { +      cmd[cmd_append_pos] = selection; +      arg.v = cmd; +      spawn(&arg); +    } +  } else if (ev->state != PropertyDelete && ev->atom == XA_WM_NAME && +             (c = getclient(ev->window)) > -1) { +    updatetitle(c); +  } +} + +void resize(int c, int w, int h) { +  XConfigureEvent ce; +  XWindowChanges wc; + +  ce.x = 0; +  ce.y = bh; +  ce.width = wc.width = w; +  ce.height = wc.height = h; +  ce.type = ConfigureNotify; +  ce.display = dpy; +  ce.event = clients[c]->win; +  ce.window = clients[c]->win; +  ce.above = None; +  ce.override_redirect = False; +  ce.border_width = 0; + +  XConfigureWindow(dpy, clients[c]->win, CWWidth | CWHeight, &wc); +  XSendEvent(dpy, clients[c]->win, False, StructureNotifyMask, (XEvent *)&ce); +} + +int resource_load(XrmDatabase db, char *name, enum resource_type rtype, +                  void *dst) { +  char **sdst = dst; +  int *idst = dst; +  float *fdst = dst; + +  char fullname[256]; +  char fullclass[256]; +  char *type; +  XrmValue ret; + +  snprintf(fullname, sizeof(fullname), "%s.%s", "tabbed", name); +  snprintf(fullclass, sizeof(fullclass), "%s.%s", "tabbed", name); +  fullname[sizeof(fullname) - 1] = fullclass[sizeof(fullclass) - 1] = '\0'; + +  XrmGetResource(db, fullname, fullclass, &type, &ret); +  if (ret.addr == NULL || strncmp("String", type, 64)) +    return 1; + +  switch (rtype) { +  case STRING: +    *sdst = ret.addr; +    break; +  case INTEGER: +    *idst = strtoul(ret.addr, NULL, 10); +    break; +  case FLOAT: +    *fdst = strtof(ret.addr, NULL); +    break; +  } +  return 0; +} + +void rotate(const Arg *arg) { +  int nsel = -1; + +  if (sel < 0) +    return; + +  if (arg->i == 0) { +    if (lastsel > -1) +      focus(lastsel); +  } else if (sel > -1) { +    /* Rotating in an arg->i step around the clients. */ +    nsel = sel + arg->i; +    while (nsel >= nclients) +      nsel -= nclients; +    while (nsel < 0) +      nsel += nclients; +    focus(nsel); +  } +} + +void run(void) { +  XEvent ev; + +  /* main event loop */ +  XSync(dpy, False); +  drawbar(); +  if (doinitspawn == True) +    spawn(NULL); + +  while (running) { +    XNextEvent(dpy, &ev); +    if (handler[ev.type]) +      (handler[ev.type])(&ev); /* call handler */ +  } +} + +void sendxembed(int c, long msg, long detail, long d1, long d2) { +  XEvent e = {0}; + +  e.xclient.window = clients[c]->win; +  e.xclient.type = ClientMessage; +  e.xclient.message_type = wmatom[XEmbed]; +  e.xclient.format = 32; +  e.xclient.data.l[0] = CurrentTime; +  e.xclient.data.l[1] = msg; +  e.xclient.data.l[2] = detail; +  e.xclient.data.l[3] = d1; +  e.xclient.data.l[4] = d2; +  XSendEvent(dpy, clients[c]->win, False, NoEventMask, &e); +} + +void setcmd(int argc, char *argv[], int replace) { +  int i; + +  cmd = emallocz((argc + 3) * sizeof(*cmd)); +  if (argc == 0) +    return; +  for (i = 0; i < argc; i++) +    cmd[i] = argv[i]; +  cmd[(replace > 0) ? replace : argc] = winid; +  cmd_append_pos = argc + !replace; +  cmd[cmd_append_pos] = cmd[cmd_append_pos + 1] = NULL; +} + +void setup(void) { +  int bitm, tx, ty, tw, th, dh, dw, isfixed; +  XClassHint class_hint; +  XSizeHints *size_hint; + +  /* clean up any zombies immediately */ +  sigchld(0); + +  /* init screen */ +  screen = DefaultScreen(dpy); +  root = RootWindow(dpy, screen); +  initfont(font); +  bh = dc.h = dc.font.height + 2; + +  /* init atoms */ +  wmatom[WMProtocols] = XInternAtom(dpy, "WM_PROTOCOLS", False); +  wmatom[WMDelete] = XInternAtom(dpy, "WM_DELETE_WINDOW", False); +  wmatom[XEmbed] = XInternAtom(dpy, "_XEMBED", False); +  wmatom[WMName] = XInternAtom(dpy, "_NET_WM_NAME", False); +  wmatom[WMState] = XInternAtom(dpy, "_NET_WM_STATE", False); +  wmatom[WMFullscreen] = XInternAtom(dpy, "_NET_WM_STATE_FULLSCREEN", False); +  wmatom[WMSelectTab] = XInternAtom(dpy, "_TABBED_SELECT_TAB", False); + +  /* init appearance */ +  wx = 0; +  wy = 0; +  ww = 800; +  wh = 600; +  isfixed = 0; + +  if (geometry) { +    tx = ty = tw = th = 0; +    bitm = XParseGeometry(geometry, &tx, &ty, (unsigned *)&tw, (unsigned *)&th); +    if (bitm & XValue) +      wx = tx; +    if (bitm & YValue) +      wy = ty; +    if (bitm & WidthValue) +      ww = tw; +    if (bitm & HeightValue) +      wh = th; +    if (bitm & XNegative && wx == 0) +      wx = -1; +    if (bitm & YNegative && wy == 0) +      wy = -1; +    if (bitm & (HeightValue | WidthValue)) +      isfixed = 1; + +    dw = DisplayWidth(dpy, screen); +    dh = DisplayHeight(dpy, screen); +    if (wx < 0) +      wx = dw + wx - ww - 1; +    if (wy < 0) +      wy = dh + wy - wh - 1; +  } + +  dc.norm[ColBG] = getcolor(normbgcolor); +  dc.norm[ColFG] = getcolor(normfgcolor); +  dc.sel[ColBG] = getcolor(selbgcolor); +  dc.sel[ColFG] = getcolor(selfgcolor); +  dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen)); +  dc.gc = XCreateGC(dpy, root, 0, 0); + +  win = XCreateSimpleWindow(dpy, root, wx, wy, ww, wh, 0, dc.norm[ColFG].pixel, +                            dc.norm[ColBG].pixel); +  XMapRaised(dpy, win); +  XSelectInput(dpy, win, +               SubstructureNotifyMask | FocusChangeMask | ButtonPressMask | +                   ExposureMask | KeyPressMask | PropertyChangeMask | +                   StructureNotifyMask | SubstructureRedirectMask); +  xerrorxlib = XSetErrorHandler(xerror); + +  class_hint.res_name = wmname; +  class_hint.res_class = "tabbed"; +  XSetClassHint(dpy, win, &class_hint); + +  size_hint = XAllocSizeHints(); +  if (!isfixed) { +    size_hint->flags = PSize; +    size_hint->height = wh; +    size_hint->width = ww; +  } else { +    size_hint->flags = PMaxSize | PMinSize; +    size_hint->min_width = size_hint->max_width = ww; +    size_hint->min_height = size_hint->max_height = wh; +  } +  XSetWMProperties(dpy, win, NULL, NULL, NULL, 0, size_hint, NULL, NULL); +  XFree(size_hint); + +  XSetWMProtocols(dpy, win, &wmatom[WMDelete], 1); + +  snprintf(winid, sizeof(winid), "%lu", win); +  setenv("XEMBED", winid, 1); + +  nextfocus = foreground; +  focus(-1); +} + +void sigchld(int unused) { +  if (signal(SIGCHLD, sigchld) == SIG_ERR) +    die("tabbed: cannot install SIGCHLD handler"); + +  while (0 < waitpid(-1, NULL, WNOHANG)) +    ; +} + +void spawn(const Arg *arg) { +  if (fork() == 0) { +    if (dpy) +      close(ConnectionNumber(dpy)); + +    setsid(); +    if (arg && arg->v) { +      execvp(((char **)arg->v)[0], (char **)arg->v); +      fprintf(stderr, "tabbed: execvp %s", ((char **)arg->v)[0]); +    } else { +      cmd[cmd_append_pos] = NULL; +      execvp(cmd[0], cmd); +      fprintf(stderr, "tabbed: execvp %s", cmd[0]); +    } +    perror(" failed"); +    exit(0); +  } +} + +int textnw(const char *text, unsigned int len) { +  XGlyphInfo ext; +  XftTextExtentsUtf8(dpy, dc.font.xfont, (XftChar8 *)text, len, &ext); +  return ext.xOff; +} + +void unmanage(int c) { +  if (c < 0 || c >= nclients) { +    drawbar(); +    XSync(dpy, False); +    return; +  } + +  if (!nclients) { +    return; +  } else if (c == 0) { +    /* First client. */ +    nclients--; +    free(clients[0]); +    memmove(&clients[0], &clients[1], sizeof(Client *) * nclients); +  } else if (c == nclients - 1) { +    /* Last client. */ +    nclients--; +    free(clients[c]); +    clients = erealloc(clients, sizeof(Client *) * nclients); +  } else { +    /* Somewhere inbetween. */ +    free(clients[c]); +    memmove(&clients[c], &clients[c + 1], +            sizeof(Client *) * (nclients - (c + 1))); +    nclients--; +  } + +  if (nclients <= 0) { +    sel = -1; +    lastsel = -1; + +    if (closelastclient) { +      running = False; +    } else if (fillagain && running) { +      spawn(NULL); +    } +  } else { +    if (c == lastsel) { +      lastsel = -1; +    } else if (lastsel > c) { +      lastsel--; +    } +    lastsel = MIN(lastsel, nclients - 1); + +    if (c == sel) { +      /* Note that focus() will never set lastsel == sel, +       * so if here lastsel == sel, it was decreased by above if() clause +       * and was actually (sel + 1) before. +       */ +      if (lastsel > 0) { +        focus(lastsel); +      } else { +        focus(0); +        lastsel = 1; +      } +    } else { +      if (sel > c) +        sel -= 1; +      if (sel >= nclients) +        sel = nclients - 1; + +      focus(sel); +    } +  } + +  drawbar(); +  XSync(dpy, False); +} + +void updatenumlockmask(void) { +  unsigned int i, j; +  XModifierKeymap *modmap; + +  numlockmask = 0; +  modmap = XGetModifierMapping(dpy); +  for (i = 0; i < 8; i++) { +    for (j = 0; j < modmap->max_keypermod; j++) { +      if (modmap->modifiermap[i * modmap->max_keypermod + j] == +          XKeysymToKeycode(dpy, XK_Num_Lock)) { +        numlockmask = (1 << i); +      } +    } +  } +  XFreeModifiermap(modmap); +} + +void updatetitle(int c) { +  if (!gettextprop(clients[c]->win, wmatom[WMName], clients[c]->name, +                   sizeof(clients[c]->name))) { +    gettextprop(clients[c]->win, XA_WM_NAME, clients[c]->name, +                sizeof(clients[c]->name)); +  } +  if (sel == c) +    xsettitle(win, clients[c]->name); +  drawbar(); +} + +/* There's no way to check accesses to destroyed windows, thus those cases are + * ignored (especially on UnmapNotify's).  Other types of errors call Xlibs + * default error handler, which may call exit.  */ +int xerror(Display *dpy, XErrorEvent *ee) { +  if (ee->error_code == BadWindow || +      (ee->request_code == X_SetInputFocus && ee->error_code == BadMatch) || +      (ee->request_code == X_PolyText8 && ee->error_code == BadDrawable) || +      (ee->request_code == X_PolyFillRectangle && +       ee->error_code == BadDrawable) || +      (ee->request_code == X_PolySegment && ee->error_code == BadDrawable) || +      (ee->request_code == X_ConfigureWindow && ee->error_code == BadMatch) || +      (ee->request_code == X_GrabButton && ee->error_code == BadAccess) || +      (ee->request_code == X_GrabKey && ee->error_code == BadAccess) || +      (ee->request_code == X_CopyArea && ee->error_code == BadDrawable)) { +    return 0; +  } + +  fprintf(stderr, "tabbed: fatal error: request code=%d, error code=%d\n", +          ee->request_code, ee->error_code); +  return xerrorxlib(dpy, ee); /* may call exit */ +} + +void xsettitle(Window w, const char *str) { +  XTextProperty xtp; + +  if (XmbTextListToTextProperty(dpy, (char **)&str, 1, XCompoundTextStyle, +                                &xtp) == Success) { +    XSetTextProperty(dpy, w, &xtp, wmatom[WMName]); +    XSetTextProperty(dpy, w, &xtp, XA_WM_NAME); +    XFree(xtp.value); +  } +} + +char *argv0; + +void usage(void) { +  die("usage: %s [-dfhsv] [-g geometry] [-n name] [-p [s+/-]pos] [-r narg] " +      "[-u color] [-U color] [-t color] [-T color] command...\n", +      argv0); +} + +int main(int argc, char *argv[]) { +  Bool detach = False; +  int replace = 0; +  char *pstr; + +  ARGBEGIN { +  case 'c': +    closelastclient = True; +    fillagain = False; +    break; +  case 'd': +    detach = True; +    break; +  case 'f': +    fillagain = True; +    break; +  case 'g': +    geometry = EARGF(usage()); +    break; +  case 'n': +    wmname = EARGF(usage()); +    break; +  case 'p': +    pstr = EARGF(usage()); +    if (pstr[0] == 's') { +      npisrelative = True; +      newposition = atoi(&pstr[1]); +    } else { +      newposition = atoi(pstr); +    } +    break; +  case 'r': +    replace = atoi(EARGF(usage())); +    break; +  case 's': +    doinitspawn = False; +    break; +  case 'v': +    die("tabbed-" VERSION ", © 2009-2012" +        " tabbed engineers, see LICENSE" +        " for details.\n"); +    break; +  case 't': +    selbgcolor = EARGF(usage()); +    break; +  case 'T': +    selfgcolor = EARGF(usage()); +    break; +  case 'u': +    normbgcolor = EARGF(usage()); +    break; +  case 'U': +    normfgcolor = EARGF(usage()); +    break; +  default: +  case 'h': +    usage(); +  } +  ARGEND; + +  if (argc < 1) { +    doinitspawn = False; +    fillagain = False; +  } + +  setcmd(argc, argv, replace); + +  if (!setlocale(LC_CTYPE, "") || !XSupportsLocale()) +    fprintf(stderr, "tabbed: no locale support\n"); +  if (!(dpy = XOpenDisplay(NULL))) +    die("tabbed: cannot open display\n"); + +  config_init(); +  setup(); +  printf("0x%lx\n", win); +  fflush(NULL); + +  if (detach) { +    if (fork() == 0) { +      fclose(stdout); +    } else { +      if (dpy) +        close(ConnectionNumber(dpy)); +      return EXIT_SUCCESS; +    } +  } + +  run(); +  cleanup(); +  XCloseDisplay(dpy); + +  return EXIT_SUCCESS; +} | 
