From d7044d310387d3752774831c1a8a768c0ca1ed67 Mon Sep 17 00:00:00 2001
From: deva <deva>
Date: Tue, 29 Mar 2011 10:04:47 +0000
Subject: Examples in docs.

---
 client/docgen/doc.h          |   8 ++
 client/docgen/docgen.cc      |  26 +++----
 client/docgen/docgen.pro     |   2 +
 client/docgen/footer.html    |   2 +
 client/docgen/generate.cc    |  81 ++++++++++++++------
 client/docgen/generate.h     |   3 +-
 client/docgen/header.html    |   9 +++
 client/docgen/highlight.cc   | 172 +++++++++++++++++++++++++++++++++++++++++++
 client/docgen/highlight.h    |  41 +++++++++++
 client/docgen/parse.cc       | 113 ++++++++++++++++++++++------
 client/docgen/style.css      | 167 +++++++++++++++++++++++++----------------
 client/widgets/altcombobox.h |   4 +-
 client/widgets/combobox.h    |   2 +-
 client/widgets/lineedit.h    |  23 +++++-
 client/widgets/metawidget.h  |   7 ++
 client/widgets/widget.h      |   5 +-
 16 files changed, 535 insertions(+), 130 deletions(-)
 create mode 100644 client/docgen/footer.html
 create mode 100644 client/docgen/header.html
 create mode 100644 client/docgen/highlight.cc
 create mode 100644 client/docgen/highlight.h

diff --git a/client/docgen/doc.h b/client/docgen/doc.h
index fc400fc..5a1d5dd 100644
--- a/client/docgen/doc.h
+++ b/client/docgen/doc.h
@@ -31,6 +31,12 @@
 #include <QVector>
 #include <QString>
 
+class Example {
+public:
+  QString name;
+  QString code;
+};
+
 class Parameter {
 public:
   QString name;
@@ -43,6 +49,7 @@ public:
   QString description;
   QString returns;
   QVector<Parameter> parameters;
+  QVector<Example> examples;
 };
 
 class Attribute {
@@ -60,6 +67,7 @@ public:
   bool screenshot;
   bool container;
   QString description;
+  QVector<Example> examples;
   QVector<Attribute> attributes;
   QVector<Method> methods;
 };
diff --git a/client/docgen/docgen.cc b/client/docgen/docgen.cc
index ee5b8f4..e4af0a3 100644
--- a/client/docgen/docgen.cc
+++ b/client/docgen/docgen.cc
@@ -44,20 +44,8 @@
 
 QString output;
 
-QString header =
-  "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.1//EN\""
-  " \"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd\">\n"
-  "<html xmlns=\"http://www.w3.org/1999/xhtml\">\n"
-  "  <head>\n"
-  "    <title>Pracro Widget Documentation</title>\n"
-  "    <meta http-equiv=\"Content-Type\" content=\"text/html\"/>\n"
-  "    <link rel=\"stylesheet\" type=\"text/css\" href=\"style.css\"/>\n"
-  "  </head> \n"
-  "  <body>\n";
-
-QString footer =
-  "  </body>\n"
-  "</html>\n";
+QString header;
+QString footer;
 
 void getInheritedAttributes(QMap<QString, QVector<Attribute> > &atts,
                             QMap<QString, Doc> &docs,
@@ -88,7 +76,7 @@ void writeDoc(QMap<QString, Doc> &docs, QString name)
   QMap<QString, QVector<Method> > meths;
   getInheritedMethods(meths, docs, doc.extends);
 
-  QString out = generate(doc, meths, atts);
+  QString out = generate(doc, meths, atts, docs);
  
   QFile::remove(output + "/" + name + ".html");
   QFile ofile(output + "/" + name + ".html");
@@ -133,6 +121,14 @@ int main(int argc, char *argv[])
 
   QMap<QString, Doc> docs;
 
+  QFile hfile("header.html");
+  hfile.open(QIODevice::ReadOnly | QIODevice::Text);
+  header = hfile.readAll();
+
+  QFile ffile("footer.html");
+  ffile.open(QIODevice::ReadOnly | QIODevice::Text);
+  footer = ffile.readAll();
+
   QDir d; d.mkdir(output);
   QFile::remove(output + "/style.css");
   QFile::copy("style.css", output + "/style.css");
diff --git a/client/docgen/docgen.pro b/client/docgen/docgen.pro
index d495439..79308b8 100644
--- a/client/docgen/docgen.pro
+++ b/client/docgen/docgen.pro
@@ -28,6 +28,7 @@ SOURCES += \
 	genimage.cc \
 	generate.cc \
 	parse.cc \
+	highlight.cc \
 	../debug.cc \
 	../lua.cc \
 	../luawidget.cc \
@@ -67,6 +68,7 @@ HEADERS += \
 	generate.h \
 	parse.h \
 	doc.h \
+	highlight.h \
 	../debug.h \
 	../lua.h \
 	../luawidget.h \
diff --git a/client/docgen/footer.html b/client/docgen/footer.html
new file mode 100644
index 0000000..b605728
--- /dev/null
+++ b/client/docgen/footer.html
@@ -0,0 +1,2 @@
+  </body>
+</html>
diff --git a/client/docgen/generate.cc b/client/docgen/generate.cc
index 5cff6b7..db7d0c9 100644
--- a/client/docgen/generate.cc
+++ b/client/docgen/generate.cc
@@ -32,6 +32,23 @@
 #include <QStringList>
 
 #include "genimage.h"
+#include "highlight.h"
+
+static QString generateExamples(QVector<Example> &examples, lang_t lang)
+{
+  QString out;
+  QVector<Example>::iterator k = examples.begin();
+  while(k != examples.end()) {
+    Example &e = *k;
+    out += "<div class=\"example\">\n";
+    out += "  <div class=\"header\"><span class=\"name\">" + e.name +
+      "</span></div>\n";
+    out += "  <pre class=\"code\">" + highlight(e.code, lang) + "</pre>";
+    out += "</div>";
+    k++;
+  }
+  return out;
+}
 
 static QString function(QString f)
 {
@@ -75,12 +92,18 @@ static QString generateMethods(QVector<Method> meths)
   QString out;
 
   foreach(Method meth, meths) {
+    QString f = meth.name;
+    QString name = f.mid(f.indexOf(" ") + 1,
+                         f.indexOf("(") - f.indexOf(" ") - 1);
+
     out += "        <div class=\"method\">\n";
-    out += "          <a name=\"" + meth.name +
+    out += "          <a name=\"" + name +
       "\"></a><div class=\"name\">" + function(meth.name) + "</div>\n";
     out += "          <div class=\"description\">" + meth.description +
       "</div>\n";
-    
+
+    out += generateExamples(meth.examples, lua);
+
     if(meth.parameters.size()) {
       out += "          <div class=\"parameters\">\n";
       foreach(Parameter parm, meth.parameters) {
@@ -150,7 +173,10 @@ static QString generateMethodOverview(Doc &doc,
 
   out += "<ul>\n";
   foreach(Method meth, meths) {
-    out += "<li><a href=\"#"+meth.name+"\">"+meth.name+"</a> " +
+    QString f = meth.name;
+    QString name = f.mid(f.indexOf(" ") + 1,
+                         f.indexOf("(") - f.indexOf(" ") - 1);
+    out += "<li><a href=\"#"+name+"\">"+meth.name+"</a> " +
       meth.description.left(meth.description.indexOf(".") + 1) + "</li>\n";
   }
   out += "</ul>\n";
@@ -158,8 +184,28 @@ static QString generateMethodOverview(Doc &doc,
   return out;
 }
 
+static QString extendedBy(QString name, QMap<QString, Doc> &docs)
+{
+  QString out;
+
+  QMap<QString, Doc>::iterator i = docs.begin();
+  while(i != docs.end()) {
+    Doc &d = i.value();
+    if(d.extends == name) {
+      if(out != "") out += ", ";
+      out += "<a href=\"" + d.name + ".html\">" + d.name + "</a>";
+    }
+    i++;
+  }
+
+  if(out != "") out = "Extended by: " + out + ".";
+
+  return out;
+}
+
 QString generate(Doc &doc, QMap<QString, QVector<Method> > &meths,
-                 QMap<QString, QVector<Attribute> > &atts)
+                 QMap<QString, QVector<Attribute> > &atts,
+                 QMap<QString, Doc> &docs)
 {
   QString out;
 
@@ -169,7 +215,8 @@ QString generate(Doc &doc, QMap<QString, QVector<Method> > &meths,
   out += "    <div class=\"doc\">\n";
   out += "      <h1>" + doc.title + "</h1>\n";
   if(doc.tag != "" ){
-    out += "      <div class=\"tagname\">&lt;" + doc.tag + " /&gt;</div>\n";
+    out += "      <div class=\"tagname\"><pre class=\"code\">" +
+      highlight("<" + doc.tag + " />", xml) + "</pre></div>\n";
   }
 
   if(doc.container) {
@@ -177,10 +224,12 @@ QString generate(Doc &doc, QMap<QString, QVector<Method> > &meths,
   }
 
   if(doc.extends != "") {
-    out += "      <div class=\"extends\">Extends: <a href=\""+doc.extends+
+    out += "      <div class=\"extends\">Extends: <a href=\"" + doc.extends +
       ".html\">" + doc.extends + "</a></div>\n";
   }
 
+  out += extendedBy(doc.name, docs);
+  
   if(doc.screenshot) {
     genImage(doc.tag);
     out += "      <div class=\"screenshot\">\n";
@@ -192,6 +241,8 @@ QString generate(Doc &doc, QMap<QString, QVector<Method> > &meths,
   out += doc.description;
   out += "      </div>\n";
 
+  out += generateExamples(doc.examples, xml);
+
   out += "      <h2>Attributes</h2>\n";
   out += "      <div class=\"attributes\">\n";
   if(doc.attributes.size()) {
@@ -232,21 +283,3 @@ QString generate(Doc &doc, QMap<QString, QVector<Method> > &meths,
 
   return out;
 }
-
-#ifdef TEST_GENERATE
-//Additional dependency files
-//deps:
-//Required cflags (autoconf vars may be used)
-//cflags:
-//Required link options (autoconf vars may be used)
-//libs:
-#include "test.h"
-
-TEST_BEGIN;
-
-// TODO: Put some testcode here (see test.h for usable macros).
-TEST_TRUE(false, "No tests yet!");
-
-TEST_END;
-
-#endif/*TEST_GENERATE*/
diff --git a/client/docgen/generate.h b/client/docgen/generate.h
index 24a3191..c706e52 100644
--- a/client/docgen/generate.h
+++ b/client/docgen/generate.h
@@ -34,6 +34,7 @@
 #include "doc.h"
 
 QString generate(Doc &doc, QMap<QString, QVector<Method> > &meths,
-                 QMap<QString, QVector<Attribute> > &atts);
+                 QMap<QString, QVector<Attribute> > &atts,
+                 QMap<QString, Doc> &docs);
 
 #endif/*__PRACRO_GENERATE_H__*/
diff --git a/client/docgen/header.html b/client/docgen/header.html
new file mode 100644
index 0000000..660e0c8
--- /dev/null
+++ b/client/docgen/header.html
@@ -0,0 +1,9 @@
+<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
+	  "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
+<html xmlns="http://www.w3.org/1999/xhtml">
+  <head>
+    <title>Pracro Widget Documentation</title>
+    <meta http-equiv="Content-Type" content="text/html"/>
+    <link rel="stylesheet" type="text/css" href="style.css"/>
+  </head>
+  <body>
diff --git a/client/docgen/highlight.cc b/client/docgen/highlight.cc
new file mode 100644
index 0000000..0857320
--- /dev/null
+++ b/client/docgen/highlight.cc
@@ -0,0 +1,172 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set et sw=2 ts=2: */
+/***************************************************************************
+ *            highlight.cc
+ *
+ *  Tue Mar 29 09:12:06 CEST 2011
+ *  Copyright 2011 Bent Bisballe Nyeng
+ *  deva@aasimon.org
+ ****************************************************************************/
+
+/*
+ *  This file is part of Pracro.
+ *
+ *  Pracro 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.
+ *
+ *  Pracro 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 Pracro; if not, write to the Free Software
+ *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA.
+ */
+#include "highlight.h"
+
+#include <QXmlStreamWriter>
+
+typedef enum {
+  none,
+  intag,
+  instring,
+  inname,
+  inattribute
+} xml_state_t;
+
+static bool isChar(QChar c)
+{
+  return
+    (c >= 'a' && c <= 'z') ||
+    (c >= 'A' && c <= 'Z') ||
+    c == '_';
+}
+ 
+static QString highlight_xml(QString code)
+{
+  QString out;
+  xml_state_t state = none;
+  bool hasname = false;
+
+  QString::iterator c = code.begin();
+  while(c != code.end()) {
+    if(*c == '<') {
+      out += "<span class=\"brackets\">&lt;</span>";
+      state = intag;
+      hasname = false;
+    } else if(state == intag && *c == '>') {
+      out += "<span class=\"brackets\">&gt;</span>";
+      state = none;
+    } else if(state == intag && *c == '/') {
+      out += "<span class=\"brackets\">/</span>";
+
+    } else if(state == intag && hasname == false && isChar(*c)) {
+      state = inname;
+      hasname = true;
+      out += "<span class=\"tag\">" + QString(*c);
+    } else if(state == inname && isChar(*c)) {
+      out += *c;
+    } else if(state == inname) {
+      out += "</span>" + QString(*c);
+      state = intag;
+
+    } else if(state == intag && hasname == true && isChar(*c)) {
+      state = inattribute;
+      out += "<span class=\"attribute\">" + QString(*c);
+    } else if(state == inattribute && isChar(*c)) {
+      out += *c;
+    } else if(state == inattribute) {
+      out += "</span>" + QString(*c);
+      state = intag;
+
+    } else if(state == intag && *c == '\"') {
+      state = instring;
+      out += "<span class=\"string\">&quot;";
+    } else if(state == instring && *c == '\"') {
+      state = intag;
+      out += "&quot;</span>";
+    } else if(state == instring) {
+      out += *c;
+
+    } else out += *c;
+    c++;
+  }
+  
+  /*
+  QXmlStreamWriter w(&e->code);
+  w.writeCharacters(line + "\n");
+  */
+
+  return out;
+}
+
+static bool isLuaChar(QChar c)
+{
+  return
+    (c >= 'a' && c <= 'z') ||
+    (c >= 'A' && c <= 'Z') ||
+    c == '_';
+}
+
+static bool isKeyword(QString code)
+{
+  return code == "if" ||
+    code == "then" ||
+    code == "else" ||
+    code == "end" ||
+    code == "this" ||
+    code == "function" ||
+    code == "local" ||
+    code == "and" ||
+    code == "or" ||
+    code == "not";
+}
+
+static QString highlight_lua(QString code)
+{
+  QString out;
+  QString word;
+
+  bool instring = false;
+
+  QString::iterator c = code.begin();
+  while(c != code.end()) {
+    if(isLuaChar(*c)) {
+      word += *c;
+    } else {
+      if(instring && *c == '\'') {
+        out += word + "'</span>";
+        instring = false;
+      } else if(instring) {
+        out += word + QString(*c);
+      } else if(isKeyword(word)) {
+        out += "<span class=\"keyword\">"+word+"</span>" + QString(*c);
+      } else if(*c == '(') {
+        out += "<span class=\"function\">"+word+"</span>" + QString(*c);
+      } else if(instring == false && *c == '\'') {
+        out += word + "<span class=\"string\">'";
+        instring = true;
+      } else {
+        out += word + QString(*c);
+      }
+      word = "";
+    }
+    
+    c++;
+  }
+
+  return out;
+}
+
+QString highlight(QString code, lang_t lang)
+{
+  switch(lang) {
+  case lua:
+    return highlight_lua(code);
+  case xml:
+    return highlight_xml(code);
+  }
+}
diff --git a/client/docgen/highlight.h b/client/docgen/highlight.h
new file mode 100644
index 0000000..e099801
--- /dev/null
+++ b/client/docgen/highlight.h
@@ -0,0 +1,41 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set et sw=2 ts=2: */
+/***************************************************************************
+ *            highlight.h
+ *
+ *  Tue Mar 29 09:12:06 CEST 2011
+ *  Copyright 2011 Bent Bisballe Nyeng
+ *  deva@aasimon.org
+ ****************************************************************************/
+
+/*
+ *  This file is part of Pracro.
+ *
+ *  Pracro 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.
+ *
+ *  Pracro 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 Pracro; if not, write to the Free Software
+ *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA.
+ */
+#ifndef __PRACRO_HIGHLIGHT_H__
+#define __PRACRO_HIGHLIGHT_H__
+
+#include <QString>
+
+typedef enum {
+  xml,
+  lua
+} lang_t;
+
+QString highlight(QString code, lang_t lang);
+
+
+#endif/*__PRACRO_HIGHLIGHT_H__*/
diff --git a/client/docgen/parse.cc b/client/docgen/parse.cc
index d1631c1..6303eae 100644
--- a/client/docgen/parse.cc
+++ b/client/docgen/parse.cc
@@ -29,16 +29,61 @@
 
 #include <QFileInfo>
 #include <QRegExp>
+#include <QStringList>
 
 typedef enum {
   none,
 	indesc,
+  indescex,
 	inatt,
 	inmeth,
+  inmethex,
 	inpar,
 	inret,
 } state_t;
 
+static QString convert(QString in)
+{
+  QString out;
+  bool linkme = false;
+  QStringList tokens = in.split(QRegExp("\\s+"));
+  QStringList::iterator i = tokens.begin();
+  while(i != tokens.end()) {
+    if(linkme) {
+      QString link = *i;
+
+      if(i->right(1) == ".") link = link.left(link.length() - 1);
+
+      if(link.indexOf("(") != -1 && link.indexOf(")") != -1 ) {
+        // method link
+        out += "<a href=\"#"+link.left(link.indexOf("("))+"\">"+link+"</a>";
+      } else {
+        // tag link
+        out += "<a href=\""+link+".html\">"+link+"</a>";
+      }
+
+      if(i->right(1) == ".") out += ".";
+      out += " ";
+
+      i++;
+      linkme = false;
+      continue;
+    }
+
+    if(*i == "@see") {
+      linkme = true;
+      i++;
+      continue;
+    }
+    
+    out += *i + " ";
+
+    i++;
+  }
+
+  return out;
+}
+
 Doc parse(QString filename)
 {
   Doc doc;
@@ -68,13 +113,14 @@ Doc parse(QString filename)
     if(line.contains("*/")) running = false;
 
     if(running) {
-      line.remove(QRegExp("^[ \t*]*"));
-      //      printf("line [%s]\n", line.toStdString().c_str());
+      line.remove(QRegExp("^[ \t]*[*][ \t]"));
       
       if(state == indesc && line.left(1) == "@") state = none;
+      if(state == indescex && line.left(1) == "@") state = none;
       if(state == inatt && line.left(1) == "@") state = none;
       if(state == inpar && line.left(1) == "@") state = inmeth;
       if(state == inret && line.left(1) == "@") state = inmeth;
+      if(state == inmethex && line.left(1) == "@") state = inmeth;
       
       if(state == none && doc.title == "") {
         doc.title = line;
@@ -83,6 +129,7 @@ Doc parse(QString filename)
       
       if(state == none && line.left(4) == "@tag") {
         doc.tag = line.right(line.length() - line.indexOf(" ") - 1);
+        doc.name = doc.tag;
         continue;
       }
 
@@ -105,12 +152,41 @@ Doc parse(QString filename)
         state = indesc;
       }
 
-
       if(state == indesc) {
         doc.description += line + "\n";
         continue;
       }
 
+      if((state == none || state == indesc || state == inmeth) 
+         && line.left(8) == "@example") {
+
+        Example e;
+        e.name = line.right(line.length() - 8);
+
+        if(state == none || state == indesc) {
+          doc.examples.push_back(e);
+          state = indescex;
+        } else {
+          Method &m = doc.methods[doc.methods.size() - 1];
+          m.examples.push_back(e);
+          state = inmethex;
+        }
+
+        continue;
+      }
+
+      if(state == indescex || state == inmethex) {
+        Example *e;
+        if(state == indescex) {
+          e = &doc.examples[doc.examples.size() - 1];
+        } else {
+          Method &m = doc.methods[doc.methods.size() - 1];
+          e = &m.examples[m.examples.size() - 1];
+        }
+        e->code += line + "\n";
+        continue;
+      }
+
       if(state == none && line.left(4) == "@att") {
         Attribute att;
         att.name = line.mid(5, line.indexOf(" ", 5) - 5);
@@ -176,23 +252,18 @@ Doc parse(QString filename)
     if(line == "/***") running = true;
   }
 
+  // Convert all @see to links
+  doc.description = convert(doc.description);
+  QVector<Attribute>::iterator i = doc.attributes.begin();
+  while(i != doc.attributes.end()) {
+    i->description = convert(i->description);
+    i++;
+  }
+  QVector<Method>::iterator j = doc.methods.begin();
+  while(j != doc.methods.end()) {
+    j->description = convert(j->description);
+    j++;
+  }
+ 
   return doc;
 }
-
-#ifdef TEST_PARSE
-//Additional dependency files
-//deps:
-//Required cflags (autoconf vars may be used)
-//cflags:
-//Required link options (autoconf vars may be used)
-//libs:
-#include "test.h"
-
-TEST_BEGIN;
-
-// TODO: Put some testcode here (see test.h for usable macros).
-TEST_TRUE(false, "No tests yet!");
-
-TEST_END;
-
-#endif/*TEST_PARSE*/
diff --git a/client/docgen/style.css b/client/docgen/style.css
index 9ef9a68..c13b0e5 100644
--- a/client/docgen/style.css
+++ b/client/docgen/style.css
@@ -1,18 +1,18 @@
 body {
-		font-family: Verdana;
-		font-size: 12px;
+    font-family: Verdana;
+    font-size: 12px;
 }
 
 a {
-		color: #5c7b8b;
+    color: #5c7b8b;
 }
 
 a:visited {
-		color: #465d69;
+    color: #465d69;
 }
 
 a:hover {
-		color: #000;
+    color: #000;
 }
 
 .doc {
@@ -20,102 +20,145 @@ a:hover {
 }
 
 .doc .title {
-		font-weight: bold;
-		font-size: 1.5em;
-		text-align: center;
+    font-weight: bold;
+    font-size: 1.5em;
+    text-align: center;
 }
 
 .doc .tagname {
-		text-align: center;
-		font-family: courier;
+    text-align: center;
 }
-
 .doc .extends {}
 .doc .container {}
 .doc .screenshot {
-		text-align: center;
+    text-align: center;
 }
 .doc .description {
-		padding-top: 4px;
-		padding-bottom: 4px;
+    padding-top: 4px;
+    padding-bottom: 4px;
+}
+
+.example {
+    border: solid #DEDEEE 1px;
+    margin-top: 4px;
+    margin-bottom: 4px;
+    border-radius: 8px 8px 8px 8px;
+}
+.example .header {
+    background-color: #D5E1E8;
+    border-radius: 8px 8px 0px 0px;
+    border: solid #84B0C7 1px;
+    padding: 4px;
+    margin: -1px;
+}
+.example .header .name {
+    border: 0px;
+    background-color: transparent;
+    padding: 0px;
+    border-radius: 0px;
+}
+.example .code {
+    padding-left: 4px;
 }
 
 .attributes {}
 .attribute {}
 .attribute .name {
-		font-weight: normal;
-		font-style: italic;
-		border: 0px;
-		border-radius: 0px 0px 0px 0px;
-		background-color: transparent;
-		display: inline;
-		color: green;
-		font-family: courier;
-		padding-right: 4px;
-		padding-left: 4px;
-		font-size: 1.2em;
+    font-weight: normal;
+    font-style: italic;
+    border: 0px;
+    border-radius: 0px 0px 0px 0px;
+    background-color: transparent;
+    display: inline;
+    color: green;
+    font-family: courier;
+    padding-right: 4px;
+    padding-left: 4px;
+    font-size: 1.2em;
 }
 .attribute .name:before {
-		content:"- ";
+    content:"- ";
 }
 .attribute .description {	
-		display: inline;
+    display: inline;
 }
 
 .attributes,
 .parameters {
-		margin-top: 8px;
-		border-top: solid #DEDEEE 1px;
-		padding-top: 8px;
+    margin-top: 8px;
+    border-top: solid #DEDEEE 1px;
+    padding-top: 8px;
 
-		margin-bottom: 8px;
-		border-bottom: solid #DEDEEE 1px;
-		padding-bottom: 8px;
+    margin-bottom: 8px;
+    border-bottom: solid #DEDEEE 1px;
+    padding-bottom: 8px;
 }
 
 .method {
-		background-color: #EEF3F5;
-		border: solid #DEDEEE 1px;
-		padding: 4px;
-		border-radius: 8px 8px 8px 8px;
-		margin-bottom: 8px;
+    background-color: #EEF3F5;
+    border: solid #DEDEEE 1px;
+    padding: 4px;
+    border-radius: 8px 8px 8px 8px;
+    margin-bottom: 8px;
 }
-
 .method .name {
-		background-color: #D5E1E8;
-		width: *;
-		border: solid #84B0C7 1px;
-		padding: 4px;
-		border-radius: 8px 8px 8px 8px;
+    background-color: #D5E1E8;
+    width: *;
+    border: solid #84B0C7 1px;
+    padding: 4px;
+    border-radius: 8px 8px 8px 8px;
 }
-
-
 .method .description {}
 .method .returns {}
+.method .luamethod {
+    font-weight: bold;
+}
+.method .luatype {
+    font-style: italic;
+    color: blue;
+}
+.method .luaparms {
+    font-style: italic;
+    color: green;
+}
 
 .parameter {}
 .parameter .name {
-		font-weight: normal;
-		font-style: italic;
-		border: 0px;
-		border-radius: 0px 0px 0px 0px;
-		background-color: transparent;
-		display: inline;
+    font-weight: normal;
+    font-style: italic;
+    border: 0px;
+    border-radius: 0px 0px 0px 0px;
+    background-color: transparent;
+    display: inline;
 }
 .parameter .description {	
-		display: inline;
+    display: inline;
 }
 
-.method .luamethod {
-    font-weight: bold;
+.code .tag {
+    color: #0000FF;
+}
+.code .attribute {
+    color: #B8840A;
+}
+.code .string {
+    color: #BC8F8F;
+}
+.code .characters {
+    color: #000000;
+}
+.code .brackets {
+    color: #000000;
 }
 
-.method .luatype {
-    font-style: italic;
-    color: blue;
+.code .keyword {
+    color: #9D20F0;
+}
+.code .function {
+    color: #0000FF;
+}
+.code .identifier {
+    color: #000000;
 }
 
-.method .luaparms {
-    font-style: italic;
-    color: green;
-}
\ No newline at end of file
+
diff --git a/client/widgets/altcombobox.h b/client/widgets/altcombobox.h
index 93ae124..ccc81cd 100644
--- a/client/widgets/altcombobox.h
+++ b/client/widgets/altcombobox.h
@@ -42,8 +42,8 @@
  * @screenshot
  * The AltComboBox is used to make a normal selection with a ComboBox but with
  * a special list item that shows an alternate widget and uses this widget for
- * input. The AltComboBox contains <code>&gt;item&lt;</code> tags in the same
- * way as the ComboBox, but may also contain an <code>&gtaltitem&lt;</code> tag
+ * input. The AltComboBox contains <code>&lt;item&gt;</code> tags in the same
+ * way as the ComboBox, but may also contain an <code>&lt;altitem&gt;</code> tag
  * which can again contain widgets.
  * @att value [altitem] The value of the item. This will be the value of the
  * AltComboBox if this item is selected.
diff --git a/client/widgets/combobox.h b/client/widgets/combobox.h
index 8a8fbd3..92a0e39 100644
--- a/client/widgets/combobox.h
+++ b/client/widgets/combobox.h
@@ -38,7 +38,7 @@
  * @extends widget
  * @screenshot
  * The ComboBox is used to make a selection from a list of items.
- * The ComboBox contains <code>&gt;item&lt;</code> tags each describing 
+ * The ComboBox contains <code>&lt;item&gt;</code> tags each describing 
  * an entry in the selection list.
  * @att type The selection method of the ComboBox. Can be one 'select', where
  * the user can select items using either the mouse or the arrow keys, 'edit',
diff --git a/client/widgets/lineedit.h b/client/widgets/lineedit.h
index 650924c..e76fe74 100644
--- a/client/widgets/lineedit.h
+++ b/client/widgets/lineedit.h
@@ -37,6 +37,11 @@
  * @extends widget
  * @screenshot
  * The lineedit is a single line textual input field.
+ * @example Simple example of usage:
+ * <lineedit name="myname" value="some text"/>
+ * @example A more advanced example:
+ * <lineedit name="myname" value="some text" readonly="true"
+ *           onChange="this:setValue('['..this:value()..']')"/>
  * @att readonly Make the lineedit readonly (not changeable by the user), but
  * still able to select and copy text.
  */
@@ -77,18 +82,31 @@ private:
 /***
  * @method nil clearSuggestions()
  * This method clears the suggestion list.
+ * For an example see @see showSuggestions().
  */
 int lin_clear_suggestions(lua_State *L);
 
 /***
  * @method nil showSuggestions()
  * This method forces showing of the suggestion list.
+ * @example Example of suggestion usage:
+ * function myOnChange()
+ *   if(isSuggested(this:value()) == false)
+ *   then
+ *     this:clearSuggestions()
+ *     this:addSuggestion(this:value() .. 'abc')
+ *     this:addSuggestion(this:value() .. 'abd')
+ *     this:addSuggestion(this:value() .. 'acd')
+ *     this:showSuggestions()
+ *   end
+ * end
  */
 int lin_show_suggestions(lua_State *L);
 
 /***
- * @method boolean isSuggested(string value, boolean joker)
+ * @method boolean isSuggested(string value)
  * This method makes a lookup in the suggestion list, searching for value.
+ * For an example see @see showSuggestions().
  * @param value A string containing the text to look for in the suggestion list.
  * @return a boolean with the value true if the string was found, false
  * if not.
@@ -99,7 +117,8 @@ int lin_is_suggested(lua_State *L);
  * @method nil addSuggestion(string suggestion)
  * Adds a suggestion to the suggestion list. The list is popped up each time 
  * a character is entered into the lineedit by the user, or it is explicitly
- * opened by calling @ref showSuggestions().
+ * opened by calling @see showSuggestions().
+ * For an example see @see showSuggestions().
  * @param suggestion A string containing the value to be added to the
  * suggestion list.
  */
diff --git a/client/widgets/metawidget.h b/client/widgets/metawidget.h
index ef3f198..ac18b91 100644
--- a/client/widgets/metawidget.h
+++ b/client/widgets/metawidget.h
@@ -42,6 +42,13 @@
  * This widget is simply a caotainer compositing its value from the widgets
  * within it. The value of the widget cannot be set and its value attribute is
  * therefore ignored.
+ * @example An example of how to use metawidget with two inner widgets:
+ * <metawidget layout="vbox" name="dims" format="${test1}: ${test2}"
+ *             storechildren="true">
+ *   <lineedit name="test1" value="test"/>
+ *   <checkbox name="test2" value="ja" truevalue="ja" falsevalue="nej"
+ *             caption="Og svaret er?"/>
+ * </metawidget>
  * @att layout the layout of the widget. Can be one of 'vbox' or 'hbox'.
  * @att storechildren This attribute indicates wether the child widgets should
  * be stored on macro commit. It can be either 'true' or 'false'.
diff --git a/client/widgets/widget.h b/client/widgets/widget.h
index ef23294..bb37c51 100644
--- a/client/widgets/widget.h
+++ b/client/widgets/widget.h
@@ -215,7 +215,8 @@ int wdg_set_visible(lua_State *L);
 
 /***
  * @method boolean valid()
- * This method is used to get the current validity state of the widget.
+ * This method is used to get the current validity state of the widget. See
+ * also @see setValid().
  * @return a boolean value. If the widget is valid the method returns true
  * otherwise it returns false.
  */
@@ -225,7 +226,7 @@ int wdg_valid(lua_State *L);
  * @method nil setValid(boolean valid)
  * This method is used to set the widgets validity state. Most widgets have a
  * visual indication of their validity state (a red background colour for
- * example) and this will also be set using this method.
+ * example) and this will also be set using this method. See also @see valid().
  * <em>NOTE</em>: An invalid widget that are not an inner widget will block a
  * server commit.
  * @param valid A boolean value. If true the widgets validity state is set to
-- 
cgit v1.2.3