From:
http://www.gnu-darwin.org/distfiles/wmtunlo-0.1.3.tar.gz
(Note that this is not really gzipped; used "tar -xvf".)
---
 wmtunlo/AUTHORS                     |   9 +
 wmtunlo/COPYING                     | 339 +++++++++++++++
 wmtunlo/Changelog                   |  12 +
 wmtunlo/Makefile                    |  23 +
 wmtunlo/README                      |  16 +
 wmtunlo/docklib.c                   | 842 ++++++++++++++++++++++++++++++++++++
 wmtunlo/docklib.h                   | 133 ++++++
 wmtunlo/examples/wmtunlorc.example1 |  25 ++
 wmtunlo/examples/wmtunlorc.example2 |  25 ++
 wmtunlo/examples/wmtunlorc.example3 |  25 ++
 wmtunlo/examples/wmtunlorc.example4 |  25 ++
 wmtunlo/examples/wmtunlorc.example5 |  25 ++
 wmtunlo/examples/wmtunlorc.example6 |  25 ++
 wmtunlo/examples/wmtunlorc.example7 |  25 ++
 wmtunlo/examples/wmtunlorc.example8 |  25 ++
 wmtunlo/examples/wmtunlorc.original |  25 ++
 wmtunlo/mask.h                      |  38 ++
 wmtunlo/master.h                    | 141 ++++++
 wmtunlo/wmtunlo.c                   | 438 +++++++++++++++++++
 19 files changed, 2216 insertions(+)
 create mode 100644 wmtunlo/AUTHORS
 create mode 100644 wmtunlo/COPYING
 create mode 100644 wmtunlo/Changelog
 create mode 100644 wmtunlo/Makefile
 create mode 100644 wmtunlo/README
 create mode 100644 wmtunlo/docklib.c
 create mode 100644 wmtunlo/docklib.h
 create mode 100644 wmtunlo/examples/wmtunlorc.example1
 create mode 100644 wmtunlo/examples/wmtunlorc.example2
 create mode 100644 wmtunlo/examples/wmtunlorc.example3
 create mode 100644 wmtunlo/examples/wmtunlorc.example4
 create mode 100644 wmtunlo/examples/wmtunlorc.example5
 create mode 100644 wmtunlo/examples/wmtunlorc.example6
 create mode 100644 wmtunlo/examples/wmtunlorc.example7
 create mode 100644 wmtunlo/examples/wmtunlorc.example8
 create mode 100644 wmtunlo/examples/wmtunlorc.original
 create mode 100644 wmtunlo/mask.h
 create mode 100644 wmtunlo/master.h
 create mode 100644 wmtunlo/wmtunlo.c

diff --git a/wmtunlo/AUTHORS b/wmtunlo/AUTHORS
new file mode 100644
index 0000000..da628b8
--- /dev/null
+++ b/wmtunlo/AUTHORS
@@ -0,0 +1,9 @@
+
+Author: 
+    Tomasz Maka <p...@ll.pl>
+
+Borrowed code:
+    wmMand-1.0
+    Mike Henderson <mghender...@lanl.gov>
+
+
diff --git a/wmtunlo/COPYING b/wmtunlo/COPYING
new file mode 100644
index 0000000..a43ea21
--- /dev/null
+++ b/wmtunlo/COPYING
@@ -0,0 +1,339 @@
+                   GNU GENERAL PUBLIC LICENSE
+                      Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.
+                          675 Mass Ave, Cambridge, MA 02139, USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                           Preamble
+
+  The licenses for most software are designed to take away your
+freedom to share and change it.  By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users.  This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it.  (Some other Free Software Foundation software is covered by
+the GNU Library General Public License instead.)  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
+this service 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 make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have.  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.
+
+  We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+  Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software.  If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+  Finally, any free program is threatened constantly by software
+patents.  We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary.  To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+                   GNU GENERAL PUBLIC LICENSE
+   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+  0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License.  The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language.  (Hereinafter, translation is included without limitation in
+the term "modification".)  Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope.  The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+  1. You may copy and distribute 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 and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+  2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+    a) You must cause the modified files to carry prominent notices
+    stating that you changed the files and the date of any change.
+
+    b) You must cause any work that you distribute or publish, that in
+    whole or in part contains or is derived from the Program or any
+    part thereof, to be licensed as a whole at no charge to all third
+    parties under the terms of this License.
+
+    c) If the modified program normally reads commands interactively
+    when run, you must cause it, when started running for such
+    interactive use in the most ordinary way, to print or display an
+    announcement including an appropriate copyright notice and a
+    notice that there is no warranty (or else, saying that you provide
+    a warranty) and that users may redistribute the program under
+    these conditions, and telling the user how to view a copy of this
+    License.  (Exception: if the Program itself is interactive but
+    does not normally print such an announcement, your work based on
+    the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole.  If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works.  But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+  3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+    a) Accompany it with the complete corresponding machine-readable
+    source code, which must be distributed under the terms of Sections
+    1 and 2 above on a medium customarily used for software interchange; or,
+
+    b) Accompany it with a written offer, valid for at least three
+    years, to give any third party, for a charge no more than your
+    cost of physically performing source distribution, a complete
+    machine-readable copy of the corresponding source code, to be
+    distributed under the terms of Sections 1 and 2 above on a medium
+    customarily used for software interchange; or,
+
+    c) Accompany it with the information you received as to the offer
+    to distribute corresponding source code.  (This alternative is
+    allowed only for noncommercial distribution and only if you
+    received the program in object code or executable form with such
+    an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it.  For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable.  However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+  4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License.  Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+  5. You are not required to accept this License, since you have not
+signed it.  However, nothing else grants you permission to modify or
+distribute the Program or its derivative works.  These actions are
+prohibited by law if you do not accept this License.  Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+  6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions.  You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+  7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+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
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all.  For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices.  Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+  8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded.  In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+  9. The Free Software Foundation may publish revised and/or new versions
+of the 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 a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation.  If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+  10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission.  For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this.  Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+                           NO WARRANTY
+
+  11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, 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.
+
+  12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE 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.
+
+                    END OF TERMS AND CONDITIONS
+
+       Appendix: 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
+convey 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) 19yy  <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 2 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, write to the Free Software
+    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+    Gnomovision version 69, Copyright (C) 19yy name of author
+    Gnomovision 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, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary.  Here is a sample; alter the names:
+
+  Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+  `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+  <signature of Ty Coon>, 1 April 1989
+  Ty Coon, President of Vice
+
+This 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 Library General
+Public License instead of this License.
diff --git a/wmtunlo/Changelog b/wmtunlo/Changelog
new file mode 100644
index 0000000..5d55040
--- /dev/null
+++ b/wmtunlo/Changelog
@@ -0,0 +1,12 @@
+
+0.1.2 (25-06-2003)
+       * command line options was replaced by a config file
+       * now, we can run up to 3 programs on double-click 
+         (based on patch by Alan Jurgensen)
+
+0.1.1 (14-04-2002)
+       * more parameters and examples
+
+0.1.0 (somewhere in the past)
+       * first release
+
diff --git a/wmtunlo/Makefile b/wmtunlo/Makefile
new file mode 100644
index 0000000..bd142e5
--- /dev/null
+++ b/wmtunlo/Makefile
@@ -0,0 +1,23 @@
+
+CFLAGS                 = -Wall -Os
+INCS                   = -I/usr/X11R6/include/X11
+LIBS                   = -L/usr/X11R6/lib -lXpm -lX11 -lXext -lm
+INSTALL_PREFIX = /usr/local
+
+
+wmtunlo: docklib.o wmtunlo.c mask.h master.h
+       gcc $(CFLAGS) $(INCS) docklib.o wmtunlo.c -o wmtunlo $(LIBS)
+
+docklib.o: docklib.c docklib.h
+       gcc $(CFLAGS) $(INCS) -c docklib.c
+
+install:
+       cp ./wmtunlo $(INSTALL_PREFIX)/bin
+
+uninstall:
+       rm -f $(INSTALL_PREFIX)/bin/wmtunlo
+
+clean:
+       rm -f *~ *.o wmtunlo
+
+
diff --git a/wmtunlo/README b/wmtunlo/README
new file mode 100644
index 0000000..722837e
--- /dev/null
+++ b/wmtunlo/README
@@ -0,0 +1,16 @@
+
+WMTUNLO 
+========
+
+It is one of the useless dockapp ever made, so use it or not....
+
+The configuration file is stored in $HOME/.clay/ directory.
+
+There are 8 examples named 'wmtunlorc.example1' thru 'wmtunlorc.example8'.
+To test these examples rename selected file to 'wmtunlorc' and put it into 
+$HOME/.clay/ directory, then run wmtunlo.
+
+You can assign three programs to run when double-click is performed (look at
+options in wmtunlorc file for details).
+
+
diff --git a/wmtunlo/docklib.c b/wmtunlo/docklib.c
new file mode 100644
index 0000000..afaf1ca
--- /dev/null
+++ b/wmtunlo/docklib.c
@@ -0,0 +1,842 @@
+/*--------------------------------*/
+/* a simple dockapp library       */
+/* made from scratch              */
+/*--------------------------------*/
+
+/* 
+       functions were written by following People:
+
+               --- linked list
+               Kresten Krab Thorup
+               Alfredo K. Kojima
+
+               --- built-in Dock module for WindowMaker
+               Alfredo K. Kojima
+
+               ---     wmgeneral (taken from wmppp)
+               Martijn Pieterse (piete...@xs4all.nl)
+
+               --- prefs routines
+               Tomasz M�ka
+*/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <ctype.h>
+#include <stdarg.h>
+#include <string.h>
+#include <unistd.h>
+#include <sys/stat.h>
+#include <sys/types.h>
+
+#include <X11/Xlib.h>
+#include <X11/xpm.h>
+#include <X11/extensions/shape.h>
+
+#include "docklib.h"
+
+/* Return a cons cell produced from (head . tail) */
+
+INLINE LinkedList* 
+list_cons(void* head, LinkedList* tail)
+{
+  LinkedList* cell;
+
+  cell = (LinkedList*)malloc(sizeof(LinkedList));
+  cell->head = head;
+  cell->tail = tail;
+  return cell;
+}
+
+/* Return the length of a list, list_length(NULL) returns zero */
+
+INLINE int
+list_length(LinkedList* list)
+{
+  int i = 0;
+  while(list)
+    {
+      i += 1;
+      list = list->tail;
+    }
+  return i;
+}
+
+/* Return the Nth element of LIST, where N count from zero.  If N 
+   larger than the list length, NULL is returned  */
+
+INLINE void*
+list_nth(int index, LinkedList* list)
+{
+  while(index-- != 0)
+    {
+      if(list->tail)
+       list = list->tail;
+      else
+       return 0;
+    }
+  return list->head;
+}
+
+/* Remove the element at the head by replacing it by its successor */
+
+INLINE void
+list_remove_head(LinkedList** list)
+{
+  if (!*list) return;  
+  if ((*list)->tail)
+    {
+      LinkedList* tail = (*list)->tail; /* fetch next */
+      *(*list) = *tail;                /* copy next to list head */
+      free(tail);                      /* free next */
+    }
+  else                         /* only one element in list */
+    {
+      free(*list);
+      (*list) = 0;
+    }
+}
+
+INLINE LinkedList *
+list_remove_elem(LinkedList* list, void* elem)
+{
+    LinkedList *tmp;
+    
+    if (list) {
+       if (list->head == elem) {
+           tmp = list->tail;
+           free(list);
+           return tmp;
+       }
+       list->tail = list_remove_elem(list->tail, elem);
+       return list;
+    }
+    return NULL;
+}
+
+
+/* Return element that has ELEM as car */
+
+INLINE LinkedList*
+list_find(LinkedList* list, void* elem)
+{
+  while(list)
+    {
+    if (list->head == elem)
+      return list;
+    list = list->tail;
+    }
+  return NULL;
+}
+
+/* Free list (backwards recursive) */
+
+INLINE void
+list_free(LinkedList* list)
+{
+  if(list)
+    {
+      list_free(list->tail);
+      free(list);
+    }
+}
+
+/* Map FUNCTION over all elements in LIST */
+
+INLINE void
+list_mapcar(LinkedList* list, void(*function)(void*))
+{
+  while(list)
+    {
+      (*function)(list->head);
+      list = list->tail;
+    }
+}
+
+
+/*
+ *----------------------------------------------------------------------
+ * parse_command--
+ *     Divides a command line into a argv/argc pair.
+ *---------------------------------------------------------------------- 
+ */
+#define PRC_ALPHA      0
+#define PRC_BLANK      1
+#define PRC_ESCAPE     2
+#define PRC_DQUOTE     3
+#define PRC_EOS                4
+#define PRC_SQUOTE     5
+
+typedef struct {
+    short nstate;
+    short output;
+} DFA;
+
+
+static DFA mtable[9][6] = {
+    {{3,1},{0,0},{4,0},{1,0},{8,0},{6,0}},
+    {{1,1},{1,1},{2,0},{3,0},{5,0},{1,1}},
+    {{1,1},{1,1},{1,1},{1,1},{5,0},{1,1}},
+    {{3,1},{5,0},{4,0},{1,0},{5,0},{6,0}},
+    {{3,1},{3,1},{3,1},{3,1},{5,0},{3,1}},
+    {{-1,-1},{0,0},{0,0},{0,0},{0,0},{0,0}}, /* final state */
+    {{6,1},{6,1},{7,0},{6,1},{5,0},{3,0}},
+    {{6,1},{6,1},{6,1},{6,1},{5,0},{6,1}},
+    {{-1,-1},{0,0},{0,0},{0,0},{0,0},{0,0}}, /* final state */
+};
+
+char*
+next_token(char *word, char **next)
+{
+    char *ptr;
+    char *ret, *t;
+    int state, ctype;
+
+    t = ret = malloc(strlen(word)+1);
+    ptr = word;
+    
+    state = 0;
+    *t = 0;
+    while (1) {
+       if (*ptr==0) 
+           ctype = PRC_EOS;
+       else if (*ptr=='\\')
+           ctype = PRC_ESCAPE;
+       else if (*ptr=='"')
+           ctype = PRC_DQUOTE;
+       else if (*ptr=='\'')
+           ctype = PRC_SQUOTE;
+       else if (*ptr==' ' || *ptr=='\t')
+           ctype = PRC_BLANK;
+       else
+           ctype = PRC_ALPHA;
+
+       if (mtable[state][ctype].output) {
+           *t = *ptr; t++;
+           *t = 0;
+       }
+       state = mtable[state][ctype].nstate;
+       ptr++;
+       if (mtable[state][0].output<0) {
+           break;
+       }
+    }
+
+    if (*ret==0)
+       t = NULL;
+    else
+       t = strdup(ret);
+
+    free(ret);
+    
+    if (ctype==PRC_EOS)
+       *next = NULL;
+    else
+       *next = ptr;
+    
+    return t;
+}
+
+
+extern void
+parse_command(char *command, char ***argv, int *argc)
+{
+    LinkedList *list = NULL;
+    char *token, *line;
+    int count, i;
+
+    line = command;
+    do {
+       token = next_token(line, &line);
+       if (token) {        
+           list = list_cons(token, list);
+       }
+    } while (token!=NULL && line!=NULL);
+
+    count = list_length(list);
+    *argv = malloc(sizeof(char*)*count);
+    i = count;
+    while (list!=NULL) {
+       (*argv)[--i] = list->head;
+       list_remove_head(&list);
+    }
+    *argc = count;
+}
+
+extern pid_t
+execCommand(char *command)
+{
+    pid_t pid;
+    char **argv;
+    int argc;
+
+    parse_command(command, &argv, &argc);
+    
+    if (argv==NULL) {
+        return 0;
+    }
+    
+    if ((pid=fork())==0) {
+        char **args;
+        int i;
+        
+        args = malloc(sizeof(char*)*(argc+1));
+        if (!args)
+          exit(10);
+        for (i=0; i<argc; i++) {
+            args[i] = argv[i];
+        }
+        args[argc] = NULL;
+        execvp(argv[0], args);
+        exit(10);
+    }
+    return pid;
+}
+
+
+
+  /*****************/
+ /* X11 Variables */
+/*****************/
+
+Window         Root;
+int                    screen;
+int                    x_fd;
+int                    d_depth;
+XSizeHints     mysizehints;
+XWMHints       mywmhints;
+Pixel          back_pix, fore_pix;
+char           *Geometry = "";
+Window         iconwin, win;
+GC                     NormalGC;
+XpmIcon                wmgen;
+Pixmap         pixmask;
+
+  /*****************/
+ /* Mouse Regions */
+/*****************/
+
+typedef struct {
+       int             enable;
+       int             top;
+       int             bottom;
+       int             left;
+       int             right;
+} MOUSE_REGION;
+
+#define MAX_MOUSE_REGION (8)
+MOUSE_REGION   mouse_region[MAX_MOUSE_REGION];
+
+  /***********************/
+ /* Function Prototypes */
+/***********************/
+
+static void GetXPM(XpmIcon *, char **);
+static Pixel GetColor(char *);
+void RedrawWindow(void);
+void AddMouseRegion(int, int, int, int, int);
+int CheckMouseRegion(int, int);
+
+/*******************************************************************************\
+|* read_rc_file                                                                
                                                                   *|
+\*******************************************************************************/
+
+void parse_rcfile(const char *filename, rckeys *keys) {
+
+       char    *p;
+       char    temp[128];
+       char    *tokens = " :\t\n";
+       FILE    *fp;
+       int             i,key;
+
+       fp = fopen(filename, "r");
+       if (fp) {
+               while (fgets(temp, 128, fp)) {
+                       key = 0;
+                       while (key >= 0 && keys[key].label) {
+                               if ((p = strstr(temp, keys[key].label))) {
+                                       p += strlen(keys[key].label);
+                                       p += strspn(p, tokens);
+                                       if ((i = strcspn(p, "#\n"))) p[i] = 0;
+                                       free(*keys[key].var);
+                                       *keys[key].var = strdup(p);
+                                       key = -1;
+                               } else key++;
+                       }
+               }
+               fclose(fp);
+       }
+}
+
+
+/*******************************************************************************\
+|* GetXPM                                                                      
                                                                   *|
+\*******************************************************************************/
+
+static void GetXPM(XpmIcon *wmgen, char *pixmap_bytes[]) {
+
+       XWindowAttributes       attributes;
+       int                                     err;
+
+       /* For the colormap */
+       XGetWindowAttributes(display, Root, &attributes);
+
+       wmgen->attributes.valuemask |= (XpmReturnPixels | XpmReturnExtensions);
+
+       err = XpmCreatePixmapFromData(display, Root, pixmap_bytes, 
&(wmgen->pixmap),
+                                       &(wmgen->mask), &(wmgen->attributes));
+       
+       if (err != XpmSuccess) {
+               fprintf(stderr, "Not enough free colorcells.\n");
+               exit(1);
+       }
+}
+
+/*******************************************************************************\
+|* GetColor                                                                    
                                                                   *|
+\*******************************************************************************/
+
+static Pixel GetColor(char *name) {
+
+       XColor                          color;
+       XWindowAttributes       attributes;
+
+       XGetWindowAttributes(display, Root, &attributes);
+
+       color.pixel = 0;
+       if (!XParseColor(display, attributes.colormap, name, &color)) {
+               fprintf(stderr, "wm.app: can't parse %s.\n", name);
+       } else if (!XAllocColor(display, attributes.colormap, &color)) {
+               fprintf(stderr, "wm.app: can't allocate %s.\n", name);
+       }
+       return color.pixel;
+}
+
+/*******************************************************************************\
+|* flush_expose                                                                
                                                                   *|
+\*******************************************************************************/
+
+static int flush_expose(Window w) {
+
+       XEvent          dummy;
+       int                     i=0;
+
+       while (XCheckTypedWindowEvent(display, w, Expose, &dummy))
+               i++;
+
+       return i;
+}
+
+/*******************************************************************************\
+|* RedrawWindow                                                                
                                                                   *|
+\*******************************************************************************/
+
+void RedrawWindow(void) {
+       
+       flush_expose(iconwin);
+       XCopyArea(display, wmgen.pixmap, iconwin, NormalGC, 
+                               0,0, wmgen.attributes.width, 
wmgen.attributes.height, 0,0);
+       flush_expose(win);
+       XCopyArea(display, wmgen.pixmap, win, NormalGC,
+                               0,0, wmgen.attributes.width, 
wmgen.attributes.height, 0,0);
+}
+
+/*******************************************************************************\
+|* RedrawWindowXY                                                              
                                                           *|
+\*******************************************************************************/
+
+void RedrawWindowXY(int x, int y) {
+       
+       flush_expose(iconwin);
+       XCopyArea(display, wmgen.pixmap, iconwin, NormalGC, 
+                               x,y, wmgen.attributes.width, 
wmgen.attributes.height, 0,0);
+       flush_expose(win);
+       XCopyArea(display, wmgen.pixmap, win, NormalGC,
+                               x,y, wmgen.attributes.width, 
wmgen.attributes.height, 0,0);
+}
+
+/*******************************************************************************\
+|* AddMouseRegion                                                              
                                                           *|
+\*******************************************************************************/
+
+void AddMouseRegion(int index, int left, int top, int right, int bottom) {
+
+       if (index < MAX_MOUSE_REGION) {
+               mouse_region[index].enable = 1;
+               mouse_region[index].top = top;
+               mouse_region[index].left = left;
+               mouse_region[index].bottom = bottom;
+               mouse_region[index].right = right;
+       }
+}
+
+/*******************************************************************************\
+|* CheckMouseRegion                                                            
                                                           *|
+\*******************************************************************************/
+
+int CheckMouseRegion(int x, int y) {
+
+       int             i;
+       int             found;
+
+       found = 0;
+
+       for (i=0; i<MAX_MOUSE_REGION && !found; i++) {
+               if (mouse_region[i].enable &&
+                       x <= mouse_region[i].right &&
+                       x >= mouse_region[i].left &&
+                       y <= mouse_region[i].bottom &&
+                       y >= mouse_region[i].top)
+                       found = 1;
+       }
+       if (!found) return -1;
+       return (i-1);
+}
+
+/*******************************************************************************\
+|* copyXPMArea                                                                 
                                                           *|
+\*******************************************************************************/
+
+void copyXPMArea(int x, int y, int sx, int sy, int dx, int dy) {
+
+       XCopyArea(display, wmgen.pixmap, wmgen.pixmap, NormalGC, x, y, sx, sy, 
dx, dy);
+
+}
+
+/*******************************************************************************\
+|* copyXBMArea                                                                 
                                                           *|
+\*******************************************************************************/
+
+void copyXBMArea(int x, int y, int sx, int sy, int dx, int dy) {
+
+       XCopyArea(display, wmgen.mask, wmgen.pixmap, NormalGC, x, y, sx, sy, 
dx, dy);
+}
+
+
+/*******************************************************************************\
+|* setMaskXY                                                                   
                                                           *|
+\*******************************************************************************/
+
+void setMaskXY(int x, int y) {
+
+        XShapeCombineMask(display, win, ShapeBounding, x, y, pixmask, 
ShapeSet);
+        XShapeCombineMask(display, iconwin, ShapeBounding, x, y, pixmask, 
ShapeSet);
+}
+
+/*******************************************************************************\
+|* openXwindow                                                                 
                                                           *|
+\*******************************************************************************/
+void openXwindow(int argc, char *argv[], char *pixmap_bytes[], char 
*pixmask_bits, int pixmask_width, int pixmask_height) {
+
+       unsigned int    borderwidth = 1;
+       XClassHint              classHint;
+       char                    *display_name = NULL;
+       char                    *wname = argv[0];
+       XTextProperty   name;
+
+       XGCValues               gcv;
+       unsigned long   gcm;
+
+
+       int                             dummy=0;
+       int                             i;
+
+       for (i=1; argv[i]; i++) {
+               if (!strcmp(argv[i], "-display")) 
+                       display_name = argv[i+1];
+       }
+
+       if (!(display = XOpenDisplay(display_name))) {
+               fprintf(stderr, "%s: can't open display %s\n", 
+                                               wname, 
XDisplayName(display_name));
+               exit(1);
+       }
+       screen  = DefaultScreen(display);
+       Root    = RootWindow(display, screen);
+       d_depth = DefaultDepth(display, screen);
+       x_fd    = XConnectionNumber(display);
+
+       /* Convert XPM to XImage */
+       GetXPM(&wmgen, pixmap_bytes);
+
+       /* Create a window to hold the stuff */
+       mysizehints.flags = USSize | USPosition;
+       mysizehints.x = 0;
+       mysizehints.y = 0;
+
+       back_pix = GetColor("white");
+       fore_pix = GetColor("black");
+
+       XWMGeometry(display, screen, Geometry, NULL, borderwidth, &mysizehints,
+                               &mysizehints.x, 
&mysizehints.y,&mysizehints.width,&mysizehints.height, &dummy);
+
+       mysizehints.width = 64;
+       mysizehints.height = 64;
+               
+       win = XCreateSimpleWindow(display, Root, mysizehints.x, mysizehints.y,
+                               mysizehints.width, mysizehints.height, 
borderwidth, fore_pix, back_pix);
+       
+       iconwin = XCreateSimpleWindow(display, win, mysizehints.x, 
mysizehints.y,
+                               mysizehints.width, mysizehints.height, 
borderwidth, fore_pix, back_pix);
+
+       /* Activate hints */
+       XSetWMNormalHints(display, win, &mysizehints);
+       classHint.res_name = wname;
+       classHint.res_class = wname;
+       XSetClassHint(display, win, &classHint);
+
+       XSelectInput(display, win, ButtonPressMask | ExposureMask | 
ButtonReleaseMask | PointerMotionMask | StructureNotifyMask);
+       XSelectInput(display, iconwin, ButtonPressMask | ExposureMask | 
ButtonReleaseMask | PointerMotionMask | StructureNotifyMask);
+
+       if (XStringListToTextProperty(&wname, 1, &name) == 0) {
+               fprintf(stderr, "%s: can't allocate window name\n", wname);
+               exit(1);
+       }
+
+       XSetWMName(display, win, &name);
+
+       /* Create GC for drawing */
+       
+       gcm = GCForeground | GCBackground | GCGraphicsExposures;
+       gcv.foreground = fore_pix;
+       gcv.background = back_pix;
+       gcv.graphics_exposures = 0;
+       NormalGC = XCreateGC(display, Root, gcm, &gcv);
+
+       /* ONLYSHAPE ON */
+
+       pixmask = XCreateBitmapFromData(display, win, pixmask_bits, 
pixmask_width, pixmask_height);
+
+       XShapeCombineMask(display, win, ShapeBounding, 0, 0, pixmask, ShapeSet);
+       XShapeCombineMask(display, iconwin, ShapeBounding, 0, 0, pixmask, 
ShapeSet);
+
+       /* ONLYSHAPE OFF */
+
+       mywmhints.initial_state = WithdrawnState;
+       mywmhints.icon_window = iconwin;
+       mywmhints.icon_x = mysizehints.x;
+       mywmhints.icon_y = mysizehints.y;
+       mywmhints.window_group = win;
+       mywmhints.flags = StateHint | IconWindowHint | IconPositionHint | 
WindowGroupHint;
+
+       XSetWMHints(display, win, &mywmhints);
+
+       XSetCommand(display, win, argv, argc);
+       XMapWindow(display, win);
+
+}
+
+
+FILE *prefs_filehandle;
+char* p_strcpy (char *dest, const char *src, int maxlength);
+char* p_strcat (char *dest, const char *src, int maxlength);
+
+/*---------------------------------------------------------------------------*/
+
+char* p_getdir_config (char *cdirectory)
+{
+static char cfgdir[MAX_PATH];
+struct stat cfg;
+
+       p_strcpy (cfgdir, getenv ("HOME"), MAX_PATH);
+       p_strcat (cfgdir, slash, MAX_PATH);
+       p_strcat (cfgdir, cdirectory, MAX_PATH);
+
+       if(stat(cfgdir, &cfg) < 0)
+               mkdir(cfgdir, S_IRUSR | S_IWUSR | S_IXUSR);
+
+       return cfgdir;
+}
+
+/*---------------------------------------------------------------------------*/
+
+char* p_getfilename_config (char *config_dir, char *config_filename)
+{
+static char filename[MAX_PATH];
+
+       p_strcpy (filename, p_getdir_config(config_dir), MAX_PATH);
+       p_strcat (filename, slash, MAX_PATH);
+       p_strcat (filename, config_filename, MAX_PATH);
+
+       return filename;
+}
+
+/*---------------------------------------------------------------------------*/
+
+void* p_prefs_openfile (char *filename, int openmode)
+{
+       prefs_filehandle = NULL;
+
+       if (openmode == P_READ)
+               prefs_filehandle = fopen (filename, "rb");
+       else if (openmode == P_WRITE)
+               prefs_filehandle = fopen (filename, "wb");
+    
+       return prefs_filehandle;
+}
+
+
+void p_prefs_closefile (void)
+{
+       fclose (prefs_filehandle);
+}
+
+/*---------------------------------------------------------------------------*/
+
+void p_prefs_put_int (char *tagname, int value)
+{
+       fprintf (prefs_filehandle, "%s=%d\n", tagname, value);
+}
+
+
+void p_prefs_put_float (char *tagname, float value)
+{
+       fprintf (prefs_filehandle, "%s=%f\n", tagname, value);
+}
+
+
+void p_prefs_put_string (char *tagname, char *value)
+{
+       fprintf (prefs_filehandle, "%s=%s\n", tagname, value);
+}
+
+
+void p_prefs_put_lf (void)
+{
+       fprintf (prefs_filehandle, "\n");
+}
+
+
+void p_prefs_put_comment (char *comment)
+{
+char text[MAX_LINE_LEN];
+
+               p_strcpy (text, "# ", MAX_LINE_LEN);
+               p_strcat (text, comment, MAX_LINE_LEN);
+               fprintf (prefs_filehandle, text);
+}
+
+/*---------------------------------------------------------------------------*/
+
+char* p_prefs_get_line_with_tag (char *tagname)
+{
+static char prfline[MAX_LINE_LEN];
+int i;
+char c;
+
+       fseek (prefs_filehandle, 0, SEEK_SET);  
+    
+       while (!feof (prefs_filehandle)) {
+               i = 0;
+
+               while (((c = fgetc (prefs_filehandle)) != crlf_char) && c!= EOF 
&& i < MAX_LINE_LEN)
+                       prfline[i++] = c;
+            
+               prfline[i] = null_char;
+        
+               if (prfline[0] != '#')
+                       if (!strncmp (tagname, prfline, strlen (tagname))) 
break;
+       }
+        
+       return prfline;
+}
+
+
+char* p_prefs_get_value_field (char *tagname)
+{
+static char valuestr[MAX_VALUE_LEN];
+char *valpos, c;
+int i;
+
+       i = 0;
+
+       if ((valpos = strchr (p_prefs_get_line_with_tag (tagname), '='))) {
+               while((c = valpos[i+1]) != null_char && i < MAX_VALUE_LEN) 
valuestr[i++] = c;
+       }
+    
+       valuestr[i] = null_char;
+       return valuestr;
+}
+
+
+int p_prefs_get_int (char *tagname)
+{
+       return (atoi (p_prefs_get_value_field (tagname)));
+}
+
+
+float p_prefs_get_float (char *tagname)
+{
+       return (atof (p_prefs_get_value_field (tagname)));
+}
+
+
+char* p_prefs_get_string (char *tagname)
+{
+       return (p_prefs_get_value_field (tagname));
+}
+
+/*---------------------------------------------------------------------------*/
+/* following functions based on samba sources.                               */
+/* safe_strcpy and safe_strcat routines written by Andrew Tridgell           */
+/*---------------------------------------------------------------------------*/
+
+char* p_strcpy (char *dest, const char *src, int maxlength)
+{
+int len;
+
+       if (!dest) {
+               printf ("ERROR: NULL dest in safe_strcpy\n");
+               return NULL;
+       }
+
+       if (!src) {
+               *dest = 0;
+               return dest;
+       }
+
+       len = strlen(src);
+
+       if (len > maxlength) {
+               printf ("ERROR: string overflow by %d in safe_strcpy [%.50s]\n",
+                               (int)(len-maxlength), src);
+               len = maxlength;
+       }
+      
+       memcpy(dest, src, len);
+       dest[len] = 0;
+       return dest;
+}
+
+/*---------------------------------------------------------------------------*/
+
+char* p_strcat (char *dest, const char *src, int maxlength)
+{
+int src_len, dest_len;
+
+       if (!dest) {
+               printf ("ERROR: NULL dest in safe_strcat\n");
+               return NULL;
+       }
+
+       if (!src) {
+               return dest;
+       }
+
+       src_len = strlen(src);
+       dest_len = strlen(dest);
+
+       if (src_len + dest_len > maxlength) {
+               printf ("ERROR: string overflow by %d in safe_strcat [%.50s]\n",
+                               (int)(src_len + dest_len - maxlength), src);
+               src_len = maxlength - dest_len;
+       }
+      
+       memcpy(&dest[dest_len], src, src_len);
+       dest[dest_len + src_len] = 0;
+       return dest;
+}
+
+/*---------------------------------------------------------------------------*/
+
+
diff --git a/wmtunlo/docklib.h b/wmtunlo/docklib.h
new file mode 100644
index 0000000..03d55f9
--- /dev/null
+++ b/wmtunlo/docklib.h
@@ -0,0 +1,133 @@
+/*--------------------------------*/
+/* a simple dockapp library       */
+/* made from scratch              */
+/*--------------------------------*/
+
+/* 
+       functions were written by following People:
+
+               --- linked list
+               Kresten Krab Thorup
+               Alfredo K. Kojima
+
+               --- built-in Dock module for WindowMaker
+               Alfredo K. Kojima
+
+               ---     wmgeneral (taken from wmppp)
+               Martijn Pieterse (piete...@xs4all.nl)
+
+               --- prefs routines
+               Tomasz M�ka
+*/
+
+
+#if defined(__GNUC__) && !defined(__STRICT_ANSI__)
+# define INLINE inline
+#else
+# define INLINE
+#endif
+
+typedef struct LinkedList {
+  void *head;
+  struct LinkedList *tail;
+} LinkedList;
+
+INLINE LinkedList* list_cons(void* head, LinkedList* tail);
+
+INLINE int list_length(LinkedList* list);
+
+INLINE void* list_nth(int index, LinkedList* list);
+
+INLINE void list_remove_head(LinkedList** list);
+
+INLINE LinkedList *list_remove_elem(LinkedList* list, void* elem);
+
+INLINE void list_mapcar(LinkedList* list, void(*function)(void*));
+
+INLINE LinkedList*list_find(LinkedList* list, void* elem);
+
+INLINE void list_free(LinkedList* list);
+
+
+extern void parse_command(char *, char ***, int *);
+
+extern pid_t execCommand(char *);
+
+
+  /***********/
+ /* Defines */
+/***********/
+
+#define MAX_MOUSE_REGION (8)
+
+  /************/
+ /* Typedefs */
+/************/
+
+typedef struct _rckeys rckeys;
+
+struct _rckeys {
+       const char      *label;
+       char            **var;
+};
+
+typedef struct {
+       Pixmap                  pixmap;
+       Pixmap                  mask;
+       XpmAttributes   attributes;
+} XpmIcon;
+
+  /*******************/
+ /* Global variable */
+/*******************/
+
+Display                *display;
+Window         Root;
+int                    d_depth;
+GC                     NormalGC;
+XpmIcon                wmgen;
+Window         iconwin, win;
+
+  /***********************/
+ /* Function Prototypes */
+/***********************/
+
+void AddMouseRegion(int index, int left, int top, int right, int bottom);
+int CheckMouseRegion(int x, int y);
+
+void openXwindow(int argc, char *argv[], char **, char *, int, int);
+void RedrawWindow(void);
+void RedrawWindowXY(int x, int y);
+
+void copyXPMArea(int, int, int, int, int, int);
+void copyXBMArea(int, int, int, int, int, int);
+void setMaskXY(int, int);
+
+void parse_rcfile(const char *, rckeys *);
+
+
+#define P_READ                 1
+#define P_WRITE                        2
+
+#define null_char              '\0'
+#define crlf_char              '\n'
+#define slash                  "/"
+
+#define MAX_LINE_LEN   512
+#define MAX_VALUE_LEN  256
+
+#define MAX_PATH               1024
+
+char* p_getfilename_config (char *config_dir, char *config_filename);
+void* p_prefs_openfile (char *filename, int openmode);
+void p_prefs_closefile (void);
+void p_prefs_put_int (char *tagname, int value);
+void p_prefs_put_float (char *tagname, float value);
+void p_prefs_put_string (char *tagname, char *value);
+void p_prefs_put_lf (void);
+void p_prefs_put_comment (char *comment);
+int p_prefs_get_int (char *tagname);
+float p_prefs_get_float (char *tagname);
+char* p_prefs_get_string (char *tagname);
+
+
diff --git a/wmtunlo/examples/wmtunlorc.example1 
b/wmtunlo/examples/wmtunlorc.example1
new file mode 100644
index 0000000..e006a44
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example1
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=1
+
+t_shade_level=10.200000
+t_persp=1200.000000
+t_mov_step=0.600000
+t_rot_step=0.300000
+
+tex_block_w=16
+tex_block_h=16
+
+t_col_add_R=24.000000
+t_col_add_G=-12.000000
+t_col_add_B=35.000000
+
+t_col_mul_R=1.000000
+t_col_mul_G=1.000000
+t_col_mul_B=0.000000
+
+t_color_1=54
+t_color_2=138
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example2 
b/wmtunlo/examples/wmtunlorc.example2
new file mode 100644
index 0000000..ec300f0
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example2
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=1
+
+t_shade_level=4.800000
+t_persp=3100.000000
+t_mov_step=2.600000
+t_rot_step=-1.100000
+
+tex_block_w=32
+tex_block_h=32
+
+t_col_add_R=2.000000
+t_col_add_G=12.000000
+t_col_add_B=35.000000
+
+t_col_mul_R=1.200000
+t_col_mul_G=0.900000
+t_col_mul_B=1.700000
+
+t_color_1=40
+t_color_2=138
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example3 
b/wmtunlo/examples/wmtunlorc.example3
new file mode 100644
index 0000000..686f94b
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example3
@@ -0,0 +1,25 @@
+shade_switch=0
+persp_switch=0
+
+t_shade_level=4.800000
+t_persp=0.000000
+t_mov_step=-0.300000
+t_rot_step=-1.100000
+
+tex_block_w=32
+tex_block_h=32
+
+t_col_add_R=7.000000
+t_col_add_G=12.000000
+t_col_add_B=35.000000
+
+t_col_mul_R=1.200000
+t_col_mul_G=0.200000
+t_col_mul_B=0.000000
+
+t_color_1=52
+t_color_2=138
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example4 
b/wmtunlo/examples/wmtunlorc.example4
new file mode 100644
index 0000000..0071128
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example4
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=0
+
+t_shade_level=22.300000
+t_persp=0.000000
+t_mov_step=0.350000
+t_rot_step=0.000000
+
+tex_block_w=8
+tex_block_h=4
+
+t_col_add_R=-22.000000
+t_col_add_G=2.000000
+t_col_add_B=1.000000
+
+t_col_mul_R=0.200000
+t_col_mul_G=0.800000
+t_col_mul_B=0.200000
+
+t_color_1=90
+t_color_2=138
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example5 
b/wmtunlo/examples/wmtunlorc.example5
new file mode 100644
index 0000000..7c016fc
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example5
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=1
+
+t_shade_level=3.800000
+t_persp=460.000000
+t_mov_step=-1.810000
+t_rot_step=-0.900000
+
+tex_block_w=64
+tex_block_h=64
+
+t_col_add_R=32.000000
+t_col_add_G=6.000000
+t_col_add_B=-19.000000
+
+t_col_mul_R=0.700000
+t_col_mul_G=1.200000
+t_col_mul_B=1.600000
+
+t_color_1=10
+t_color_2=138
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example6 
b/wmtunlo/examples/wmtunlorc.example6
new file mode 100644
index 0000000..792c70d
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example6
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=1
+
+t_shade_level=3.300000
+t_persp=5112.000000
+t_mov_step=-1.810000
+t_rot_step=-2.200000
+
+tex_block_w=16
+tex_block_h=16
+
+t_col_add_R=2.000000
+t_col_add_G=19.000000
+t_col_add_B=-0.600000
+
+t_col_mul_R=3.100000
+t_col_mul_G=1.000000
+t_col_mul_B=0.200000
+
+t_color_1=82
+t_color_2=140
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example7 
b/wmtunlo/examples/wmtunlorc.example7
new file mode 100644
index 0000000..a37ccf9
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example7
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=0
+
+t_shade_level=7.100000
+t_persp=5112.000000
+t_mov_step=-1.810000
+t_rot_step=-1.100000
+
+tex_block_w=16
+tex_block_h=16
+
+t_col_add_R=-0.300000
+t_col_add_G=7.000000
+t_col_add_B=0.200000
+
+t_col_mul_R=3.100000
+t_col_mul_G=0.410000
+t_col_mul_B=2.200000
+
+t_color_1=82
+t_color_2=180
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.example8 
b/wmtunlo/examples/wmtunlorc.example8
new file mode 100644
index 0000000..ab057a9
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.example8
@@ -0,0 +1,25 @@
+shade_switch=0
+persp_switch=0
+
+t_shade_level=0.000000
+t_persp=0.000000
+t_mov_step=0.00000
+t_rot_step=2.300000
+
+tex_block_w=16
+tex_block_h=16
+
+t_col_add_R=-2.300000
+t_col_add_G=6.000000
+t_col_add_B=8.200000
+
+t_col_mul_R=3.100000
+t_col_mul_G=7.410000
+t_col_mul_B=9.200000
+
+t_color_1=12
+t_color_2=40
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/examples/wmtunlorc.original 
b/wmtunlo/examples/wmtunlorc.original
new file mode 100644
index 0000000..f0434d6
--- /dev/null
+++ b/wmtunlo/examples/wmtunlorc.original
@@ -0,0 +1,25 @@
+shade_switch=1
+persp_switch=1
+
+t_shade_level=5.200000
+t_persp=1000.000000
+t_mov_step=0.600000
+t_rot_step=0.300000
+
+tex_block_w=32
+tex_block_h=32
+
+t_col_add_R=24.000000
+t_col_add_G=-12.000000
+t_col_add_B=55.000000
+
+t_col_mul_R=1.000000
+t_col_mul_G=1.000000
+t_col_mul_B=0.000000
+
+t_color_1=20
+t_color_2=128
+
+command1=xlock
+command2=xlock -mode thornbird
+command3=xlock -mode blank
diff --git a/wmtunlo/mask.h b/wmtunlo/mask.h
new file mode 100644
index 0000000..0851918
--- /dev/null
+++ b/wmtunlo/mask.h
@@ -0,0 +1,38 @@
+#define dock_mask_width 64
+#define dock_mask_height 64
+static char dock_mask_bits[] = {
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,
+ 0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,
+ 0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,
+ 0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,
+ 0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,
+ 0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,
+ 0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,
+ 0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,
+ 0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,
+ 0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,
+ 0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,
+ 0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,
+ 0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,
+ 0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,
+ 0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,
+ 0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,
+ 0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,
+ 0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,
+ 0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,
+ 0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,
+ 0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,
+ 0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,
+ 0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,
+ 0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,
+ 0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,
+ 0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0x0f,0xf0,0xff,0xff,0xff,0xff,0xff,0xff,0x0f,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00};
diff --git a/wmtunlo/master.h b/wmtunlo/master.h
new file mode 100644
index 0000000..e32e405
--- /dev/null
+++ b/wmtunlo/master.h
@@ -0,0 +1,141 @@
+/* XPM */
+static char * dock_master[] = {
+"64 64 74 1",
+"      c None",
+".     c #F7FBFF",
+"+     c #F7F7FF",
+"@     c #EFF7FF",
+"#     c #EFF3FF",
+"$     c #EFEFFF",
+"%     c #EBF4FC",
+"&     c #EBEFFC",
+"*     c #EBEBFC",
+"=     c #E2EBF3",
+"-     c #E2E6F3",
+";     c #E2E2F3",
+">     c #E3E3EB",
+",     c #DAE3EB",
+"'     c #DADEEB",
+")     c #DADAEB",
+"!     c #DBDBE3",
+"~     c #D2DBE3",
+"{     c #D2D6E3",
+"]     c #A8ACB9",
+"^     c #000000",
+"/     c #545865",
+"(     c #A8A8B9",
+"_     c #9FA8B0",
+":     c #9FA3B0",
+"<     c #9F9FB0",
+"[     c #97A0A8",
+"}     c #979BA8",
+"|     c #9797A8",
+"1     c #97979F",
+"2     c #8E979F",
+"3     c #8E929F",
+"4     c #8E8E9F",
+"5     c #8F8F97",
+"6     c #868F97",
+"7     c #868A97",
+"8     c #868697",
+"9     c #7E878F",
+"0     c #7E828F",
+"a     c #191919",
+"b     c #DACDC9",
+"c     c #7E7E8F",
+"d     c #545465",
+"e     c #757E86",
+"f     c #4B545C",
+"g     c #757986",
+"h     c #4B4F5C",
+"i     c #757586",
+"j     c #4B4B5C",
+"k     c #6D767E",
+"l     c #4C4C54",
+"m     c #434C54",
+"n     c #6D717E",
+"o     c #434754",
+"p     c #6D6D7E",
+"q     c #434354",
+"r     c #6E6E76",
+"s     c #44444C",
+"t     c #656E76",
+"u     c #3B444C",
+"v     c #656976",
+"w     c #3B3F4C",
+"x     c #656576",
+"y     c #65656D",
+"z     c #3B3B4C",
+"A     c #5C656D",
+"B     c #323B43",
+"C     c #5C606D",
+"D     c #323643",
+"E     c #5C5C6D",
+"F     c #323243",
+"G     c #545D65",
+"H     c #2A333B",
+"I     c #2A2E3B",
+"..+++@@@#######$$%%%%&&&&&&**====-------;>,,,,'''''')!~~~~~{{{]^",
+".+++@@@#######$$%%%%&&&&&&**====-------;>,,,,'''''')!~~~~~{{{{/^",
+"++((((__::::::<<<<[[}}}}}}}|||12233333344456667777778889990000/^",
+"++(((__::::::<<<<[[}}}}}}}|||122333333444566677777788899900000/^",
+"+@((^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^00/^",
+"@@(_^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaab0c/^",
+"@@__^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabccd^",
+"@#_:^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabccd^",
+"##::^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabced^",
+"##::^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabeef^",
+"##::^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabeef^",
+"##::^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabeef^",
+"##::^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabegh^",
+"##:<^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabggh^",
+"#$<<^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabggh^",
+"$$<<^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabggh^",
+"$%<<^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabggh^",
+"%%<[^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabggh^",
+"%%[[^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabgij^",
+"%%[}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabiij^",
+"%&}}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabikj^",
+"&&}}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabkkl^",
+"&&}}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabkkm^",
+"&&}}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabkkm^",
+"&&}}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabknm^",
+"&&}}^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabnno^",
+"&*}|^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabnno^",
+"**||^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabnno^",
+"*=||^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabnno^",
+"==|1^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabnno^",
+"==12^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabnpo^",
+"==22^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabppq^",
+"=-23^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabprq^",
+"--33^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabrrs^",
+"--33^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabrtu^",
+"--33^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabttu^",
+"--33^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabttu^",
+"--33^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabtvw^",
+"--34^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabvvw^",
+"-;44^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabvvw^",
+";>44^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabvvw^",
+">,45^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabvvw^",
+",,56^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabvvw^",
+",,66^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabvxw^",
+",,66^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabxyz^",
+",'67^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabyyz^",
+"''77^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabyAB^",
+"''77^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabAAB^",
+"''77^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabAAB^",
+"''77^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabACB^",
+"''77^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCCD^",
+"')78^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCCD^",
+")!88^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCCD^",
+"!~88^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCCD^",
+"~~89^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCCD^",
+"~~99^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCCD^",
+"~~99^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabCEF^",
+"~~90^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabEEF^",
+"~{00^aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaabEGH^",
+"{{00bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbGGH^",
+"{{0000ccceeeeggggggiikkkknnnnnnpprrtttvvvvvvxyyAAACCCCCCCEEGGGH^",
+"{{000ccceeeeggggggiikkkknnnnnnpprrtttvvvvvvxyyAAACCCCCCCEEGGGGH^",
+"]/////dddfffhhhhhhjjjlmmmooooooqqsuuuwwwwwwwzzBBBBDDDDDDFFHHHHI^",
+"^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^"};
diff --git a/wmtunlo/wmtunlo.c b/wmtunlo/wmtunlo.c
new file mode 100644
index 0000000..17ba29f
--- /dev/null
+++ b/wmtunlo/wmtunlo.c
@@ -0,0 +1,438 @@
+
+/*
+ * wmtunlo
+ *
+ * Copyright (C) 2001 pasp
+ *
+ * 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 2 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, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+
+#include <stdio.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <string.h>
+#include <time.h>
+#include <X11/X.h>
+#include <X11/xpm.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <math.h>
+
+#include <X11/Xlib.h>
+#include <X11/xpm.h>
+#include <X11/extensions/shape.h>
+
+#include "master.h"
+#include "mask.h"
+
+#include "docklib.h"
+
+void wmtunlo_read_prefs(void);
+void wmtunlo_write_prefs(void);
+
+void   ButtonPressEvent(XButtonEvent *);
+int    GotFirstClick1, GotDoubleClick1;
+int    GotFirstClick2, GotDoubleClick2;
+int    GotFirstClick3, GotDoubleClick3;
+int    DblClkDelay;
+int    HasExecute;
+char   ClickCommand1[MAX_PATH];
+char   ClickCommand2[MAX_PATH];
+char   ClickCommand3[MAX_PATH];
+
+int            tex_block_w, tex_block_h, shade_switch, persp_switch;
+int            t_color_1, t_color_2;
+float  t_persp, t_shade_level;
+float  t_mov_counter, t_rot_counter;
+float  t_mov_step, t_rot_step;
+float  t_col_add_R, t_col_add_G, t_col_add_B;
+float  t_col_mul_R, t_col_mul_G, t_col_mul_B;
+
+int main(int argc, char *argv[])
+{
+
+typedef struct {
+    Display                    *display;
+    int                                screen;
+    Visual                     *visual;
+    int                                depth;
+    Colormap           cmap;
+    int                                format;
+    int                                bitmap_pad;
+    int                                Color[256];
+    unsigned char      R[256];
+    unsigned char      G[256];
+    unsigned char      B[256];
+} DisplayInfo;
+
+XImage *xim;
+XEvent event;
+XColor xColor, xColors[256];
+DisplayInfo Info;
+int i, j, shade, k, l, p, q, result, col;
+float xz, yz, px, py;
+unsigned char *Image, *Texture, *Tunnel_map;
+
+
+    /* Read preferences */
+       wmtunlo_read_prefs(); 
+
+    /* Open window */
+    openXwindow(argc, argv, dock_master, dock_mask_bits, dock_mask_width, 
dock_mask_height);
+
+    /* Get Display parameters */
+    Info.display = display;
+    Info.screen  = DefaultScreen(display);
+    Info.visual  = DefaultVisual(display, Info.screen);
+    Info.depth   = DefaultDepth(display, Info.screen);
+    Info.cmap    = DefaultColormap(display, 0);
+
+    /* Initialize Color Table */
+
+               for (i=0; i<256; ++i) {
+                       col = t_col_mul_R * (float)i + t_col_add_R;
+                       if(col<0) col = 0;
+                       if(col>255) col = 255;
+                       Info.R[i] = col;
+
+                       col = t_col_mul_G * (float)i + t_col_add_G;
+                       if(col<0) col = 0;
+                       if(col>255) col = 255;
+                       Info.G[i] = col;
+
+                       col = t_col_mul_B * (float)i + t_col_add_B;
+                       if(col<0) col = 0;
+                       if(col>255) col = 255;
+                       Info.B[i] = col;
+               }
+
+       /* Create an XImage with null data. Then allocate space for data. */
+    Info.format = ZPixmap;
+    if (Info.depth == 8){
+
+        Info.bitmap_pad = 8;
+
+               /* Set a private colormap */
+               Info.cmap = XCreateColormap(Info.display, 
RootWindow(Info.display, Info.screen), Info.visual, AllocAll);
+               for (i=0; i<256; ++i){
+                           Info.Color[i] = i;
+                           xColors[i].pixel = i;
+                       xColors[i].red   = (unsigned short)Info.R[i] << 8;
+                           xColors[i].green = (unsigned short)Info.G[i] << 8;
+                           xColors[i].blue  = (unsigned short)Info.B[i] << 8;
+                           xColors[i].flags = DoRed | DoGreen | DoBlue;
+           }
+               XStoreColors(Info.display, Info.cmap, xColors, 256);
+               XSetWindowColormap(Info.display, win, Info.cmap);
+
+    } else if (Info.depth > 8) {
+
+               /* Allocate Colors */
+        for (i=0; i<256; ++i){
+                   xColor.red   = (unsigned short)Info.R[i] << 8;
+                   xColor.green = (unsigned short)Info.G[i] << 8;
+                   xColor.blue  = (unsigned short)Info.B[i] << 8;
+                   xColor.flags = DoRed | DoGreen | DoBlue;
+                   XAllocColor(Info.display, Info.cmap, &xColor);
+                   Info.Color[i] = xColor.pixel;
+        }
+        Info.bitmap_pad = 32;
+
+    } else {
+               fprintf(stderr, "Need at least 8-bit display!\n");
+               exit(-1);
+    }
+
+    xim = XCreateImage(Info.display, Info.visual, Info.depth, Info.format, 0, 
(char *)0, 54, 54, Info.bitmap_pad, 0);
+    xim->data = (char *)malloc(xim->bytes_per_line * 54 );
+
+    /* Allocate memory for image data */
+    Image    = (unsigned char *)malloc(sizeof(unsigned char)*54*54);
+
+       /* Generate tunnel map */
+    Tunnel_map = (unsigned char *)malloc(sizeof(unsigned char)*54*54*4);
+
+               for (k=0, yz=-54/2; yz<54/2; yz++)
+                  for (xz=-54/2; xz<54/2; xz++) {
+
+                       py = sqrt(xz*xz + yz*yz);
+                       shade= py * t_shade_level;
+                               if(shade>255) shade=255;
+                       if(persp_switch) py = t_persp/py;
+                       px = atan2(yz,xz) * 128.0 / M_PI;
+                       result = (int)py * 256 + (int)px;
+
+                       Tunnel_map[k++]= result&0xff; 
+                               Tunnel_map[k++]= result>>8;
+                       Tunnel_map[k++]= shade&0xff;
+                   }
+
+       /* Generate texture */
+    Texture = (unsigned char *)malloc(sizeof(unsigned char)*256*256);
+
+       for(i=0,q=1;i<256;i+=tex_block_h, q*=-1)
+               for(j=0,p=q;j<256;j+=tex_block_w, p*=-1) 
+               for(l=0;l<tex_block_h;l++)
+                       for(k=0;k<tex_block_w;k++)
+                                       if((j+k < 256) && (i+l < 256)) {
+                                               if(p>0)
+                                                       Texture[(i+l)*256+j+k] 
= t_color_1;
+                                               else 
+                                                       Texture[(i+l)*256+j+k] 
= t_color_2;
+                                               }
+
+       /* Animation. */
+
+       t_mov_counter = 0.0;
+       t_rot_counter = 0.0;
+
+    while(1) {
+
+               /* Process any pending X events. */
+
+        while(XPending(display)){
+            XNextEvent(display, &event);
+            switch(event.type){
+                case Expose:
+                        RedrawWindow();
+                        break;
+                case EnterNotify:
+                                               XSetInputFocus(display, 
iconwin, RevertToNone, CurrentTime);
+                                               if (Info.depth == 8) 
XInstallColormap(display, Info.cmap);
+                        break;
+                case LeaveNotify:
+                                               XSetInputFocus(display, None, 
RevertToNone, CurrentTime);
+                                               if (Info.depth == 8) 
XUninstallColormap(display, Info.cmap);
+                        break;
+                               case ButtonPress:
+                                               
ButtonPressEvent(&event.xbutton);
+                                               break;
+                               case ButtonRelease:
+                                               break;
+               }
+        }
+
+               /* Draw tunnel. */
+
+               for(i=0;i<54;i++)
+                       for(j=0;j<54;j++) {
+
+                               if(shade_switch)
+                           *(Image + 54*i + j) = (Tunnel_map[(54*i+j)*3 + 2] * 
Texture[ \
+                                       ((Tunnel_map[(54*i+j)*3 + 1] + 
(int)t_mov_counter) & 0xff)*256 + \
+                                       Tunnel_map[(54*i+j)*3 + 0] - 
(int)t_rot_counter]) >> 8;
+                               else
+                           *(Image + 54*i + j) = Texture[ \
+                                       ((Tunnel_map[(54*i+j)*3 + 1] + 
(int)t_mov_counter) & 0xff)*256 + \
+                                       Tunnel_map[(54*i+j)*3 + 0] - 
(int)t_rot_counter];
+                       }
+
+               /* Move tunnel. */
+
+               t_mov_counter += t_mov_step;
+               if(t_mov_counter > 256.0) t_mov_counter = 0.0;
+               if(t_mov_counter < 0.0) t_mov_counter = 256.0;
+               t_rot_counter += t_rot_step;
+               if(t_rot_counter > 256.0) t_rot_counter = 0.0;
+               if(t_rot_counter < 0.0) t_rot_counter = 256.0;
+
+               /* Paste up image. */
+           for ( i=0; i<54; ++i )
+                       for ( j=0; j<54; ++j ) {
+                           XPutPixel(xim, i, j,  Info.Color[*(Image + j*54 + 
i)]);
+                           XFlush(display);
+                       }
+
+           XPutImage(display, wmgen.pixmap, NormalGC, xim, 0, 0, 5, 5, 54, 54);
+
+               /* Make changes visible */
+           RedrawWindow();
+
+               /* Wait for next update  */
+               usleep(10000L);
+    }
+
+    free(xim->data);
+    XDestroyImage(xim);
+
+       free(Image);
+       free(Tunnel_map);
+       free(Texture);
+}
+
+/*
+ *  This routine handles button presses.
+ *
+ *   Double click on
+ *              Mouse Button 1: Execute the command defined in the -e 
command-line option.
+ *              Mouse Button 2: No action assigned.
+ *              Mouse Button 3: No action assigned.
+ *
+ *
+ */
+void ButtonPressEvent(XButtonEvent *xev){
+
+    DblClkDelay = 0;
+    if ((xev->button == Button1) && (xev->type == ButtonPress)){
+        if (GotFirstClick1) GotDoubleClick1 = 1;
+        else GotFirstClick1 = 1;
+    } else if ((xev->button == Button2) && (xev->type == ButtonPress)){
+        if (GotFirstClick2) GotDoubleClick2 = 1;
+        else GotFirstClick2 = 1;
+    } else if ((xev->button == Button3) && (xev->type == ButtonPress)){
+        if (GotFirstClick3) GotDoubleClick3 = 1;
+        else GotFirstClick3 = 1;
+    }
+
+    /*
+     *  We got a double click on Mouse Button1 (i.e. the left one)
+     */
+    if (GotDoubleClick1) {
+        GotFirstClick1 = 0;
+        GotDoubleClick1 = 0;
+        system(ClickCommand1);
+    }
+
+    /*
+     *  We got a double click on Mouse Button2 (i.e. the left one)
+     */
+    if (GotDoubleClick2) {
+        GotFirstClick2 = 0;
+        GotDoubleClick2 = 0;
+        system(ClickCommand2);
+    }
+
+    /*
+     *  We got a double click on Mouse Button3 (i.e. the left one)
+     */
+    if (GotDoubleClick3) {
+        GotFirstClick3 = 0;
+        GotDoubleClick3 = 0;
+        system(ClickCommand3);
+    }
+
+   return;
+
+}
+
+/*----------------------------------------------------------------------*/
+/* Write preferences */
+
+void wmtunlo_write_prefs(void) 
+{
+       if (p_prefs_openfile (p_getfilename_config (".clay", "wmtunlorc"), 
P_WRITE)) {
+
+               p_prefs_put_int("shade_switch", shade_switch);
+               p_prefs_put_int("persp_switch", persp_switch);
+               p_prefs_put_lf ();
+               p_prefs_put_float("t_shade_level", t_shade_level);
+               p_prefs_put_float("t_persp", t_persp);
+               p_prefs_put_float("t_mov_step", t_mov_step);
+               p_prefs_put_float("t_rot_step", t_rot_step);
+               p_prefs_put_lf ();
+               p_prefs_put_int("tex_block_w", tex_block_w);
+               p_prefs_put_int("tex_block_h", tex_block_h);
+               p_prefs_put_lf ();
+               p_prefs_put_float("t_col_add_R", t_col_add_R);
+               p_prefs_put_float("t_col_add_G", t_col_add_G);
+               p_prefs_put_float("t_col_add_B", t_col_add_B);
+               p_prefs_put_lf ();
+               p_prefs_put_float("t_col_mul_R", t_col_mul_R);
+               p_prefs_put_float("t_col_mul_G", t_col_mul_G);
+               p_prefs_put_float("t_col_mul_B", t_col_mul_B);
+               p_prefs_put_lf ();
+               p_prefs_put_int("t_color_1", t_color_1);
+               p_prefs_put_int("t_color_2", t_color_2);
+               p_prefs_put_lf ();
+               p_prefs_put_string("command1", ClickCommand1);
+               p_prefs_put_string("command2", ClickCommand2);
+               p_prefs_put_string("command3", ClickCommand3);
+
+       }
+        
+       p_prefs_closefile ();
+}
+
+/*----------------------------------------------------------------------*/
+/* Read preferences */
+
+void wmtunlo_read_prefs(void) 
+{
+       if (p_prefs_openfile (p_getfilename_config(".clay", "wmtunlorc"), 
P_READ)) {
+
+               shade_switch = p_prefs_get_int("shade_switch");
+               persp_switch = p_prefs_get_int("persp_switch");
+
+               t_shade_level = p_prefs_get_float("t_shade_level");
+               t_persp = p_prefs_get_float("t_persp");
+               t_mov_step = p_prefs_get_float("t_mov_step");
+               t_rot_step = p_prefs_get_float("t_rot_step");
+
+               tex_block_w = p_prefs_get_int("tex_block_w");
+               tex_block_h = p_prefs_get_int("tex_block_h");
+
+               t_col_add_R = p_prefs_get_float("t_col_add_R");
+               t_col_add_G = p_prefs_get_float("t_col_add_G");
+               t_col_add_B = p_prefs_get_float("t_col_add_B");
+
+               t_col_mul_R = p_prefs_get_float("t_col_mul_R");
+               t_col_mul_G = p_prefs_get_float("t_col_mul_G");
+               t_col_mul_B = p_prefs_get_float("t_col_mul_B");
+
+               t_color_1 = p_prefs_get_int("t_color_1") % 255;
+               t_color_2 = p_prefs_get_int("t_color_2") % 255;
+
+               strcpy(ClickCommand1, p_prefs_get_string ("command1"));
+               strcpy(ClickCommand2, p_prefs_get_string ("command2"));
+               strcpy(ClickCommand3, p_prefs_get_string ("command3"));
+    
+               p_prefs_closefile ();
+        
+       } else {
+
+               shade_switch = 1;
+               persp_switch = 1;
+
+               t_shade_level = 5.2;
+               t_persp = 1000.0;
+               t_mov_step = 0.6;
+               t_rot_step = 0.3;
+
+               tex_block_w = 32;
+               tex_block_h = 32;
+
+               t_col_add_R = 24.0;
+               t_col_add_G = -12.0;
+               t_col_add_B = 55.0;
+
+               t_col_mul_R = 1.0;
+               t_col_mul_G = 1.0;
+               t_col_mul_B = 0.0;
+
+               t_color_1 = 20;
+               t_color_2 = 128;
+
+               strcpy(ClickCommand1, "xlock");
+               strcpy(ClickCommand2, "xlock -mode thornbird");
+               strcpy(ClickCommand3, "xlock -mode blank");
+
+               wmtunlo_write_prefs ();
+
+       }
+}
+
+/*----------------------------------------------------------------------*/
+
-- 
2.7.4


-- 
To unsubscribe, send mail to wmaker-dev-unsubscr...@lists.windowmaker.org.

Reply via email to