Implemented syslink control.

diff --git a/dlls/comctl32/Makefile.in b/dlls/comctl32/Makefile.in
index 87405ab..2218bd2 100644
--- a/dlls/comctl32/Makefile.in
+++ b/dlls/comctl32/Makefile.in
@@ -30,6 +30,7 @@
 	smoothscroll.c \
 	string.c \
 	status.c \
+	syslink.c \
 	tab.c \
 	toolbar.c \
 	tooltips.c \
diff --git a/dlls/comctl32/commctrl.c b/dlls/comctl32/commctrl.c
index a2af02a..12ed0e6 100644
--- a/dlls/comctl32/commctrl.c
+++ b/dlls/comctl32/commctrl.c
@@ -101,6 +101,8 @@
 extern void REBAR_Unregister(void);
 extern void STATUS_Register(void);
 extern void STATUS_Unregister(void);
+extern void SYSLINK_Register(void);
+extern void SYSLINK_Unregister(void);
 extern void TAB_Register(void);
 extern void TAB_Unregister(void);
 extern void TOOLBAR_Register(void);
@@ -174,6 +176,7 @@
             LISTVIEW_Register ();
             PROGRESS_Register ();
             STATUS_Register ();
+            SYSLINK_Register ();
             TAB_Register ();
             TOOLBAR_Register ();
             TOOLTIPS_Register ();
@@ -198,6 +201,7 @@
             PROGRESS_Unregister ();
             REBAR_Unregister ();
             STATUS_Unregister ();
+            SYSLINK_Unregister ();
             TAB_Unregister ();
             TOOLBAR_Unregister ();
             TOOLTIPS_Unregister ();
@@ -711,6 +715,10 @@
 		NATIVEFONT_Register ();
 		break;
 
+	    case ICC_LINK_CLASS:
+		SYSLINK_Register ();
+		break;
+
 	    default:
 		FIXME("Unknown class! dwICC=0x%lX\n", dwMask);
 		break;
diff --git a/dlls/comctl32/syslink.c b/dlls/comctl32/syslink.c
new file mode 100644
index 0000000..fa396fc
--- /dev/null
+++ b/dlls/comctl32/syslink.c
@@ -0,0 +1,1682 @@
+/*
+ * SysLink control
+ *
+ * Copyright 2004 Thomas Weidenmueller <w3seek@reactos.com>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+ * TODO:
+ * - Fix SHIFT+TAB and TAB issue (wrong link is selected when control gets the focus)
+ * - Better string parsing
+ * - Improve word wrapping
+ * - Control styles?!
+ *
+ */
+
+#include <stdarg.h>
+#include <string.h>
+#include "windef.h"
+#include "winbase.h"
+#include "wingdi.h"
+#include "winuser.h"
+#include "winnls.h"
+#include "commctrl.h"
+#include "comctl32.h"
+#include "wine/debug.h"
+
+WINE_DEFAULT_DEBUG_CHANNEL(progress);
+
+INT WINAPI StrCmpNIW(LPCWSTR,LPCWSTR,INT);
+
+#define SYSLINK_Alloc(size)        HeapAlloc(GetProcessHeap(), 0, (size))
+#define SYSLINK_Free(ptr)          HeapFree(GetProcessHeap(), 0, (ptr))
+#define SYSLINK_ReAlloc(ptr, size) HeapReAlloc(GetProcessHeap(), 0, ptr, (size))
+
+typedef struct
+{
+    int nChars;
+    RECT rc;
+} DOC_TEXTBLOCK, *PDOC_TEXTBLOCK;
+
+#define LIF_FLAGSMASK   (LIF_STATE | LIF_ITEMID | LIF_URL)
+#define LIS_MASK        (LIS_FOCUSED | LIS_ENABLED | LIS_VISITED)
+
+typedef enum
+{
+    slText = 0,
+    slLink
+} SL_ITEM_TYPE;
+
+typedef struct _DOC_ITEM
+{
+    struct _DOC_ITEM *Next; /* Address to the next item */
+    LPWSTR Text;            /* Text of the document item */
+    UINT nText;             /* Number of characters of the text */
+    SL_ITEM_TYPE Type;      /* type of the item */
+    PDOC_TEXTBLOCK Blocks;  /* Array of text blocks */
+    union
+    {
+        struct
+        {
+            UINT state;     /* Link state */
+            WCHAR *szID;    /* Link ID string */
+            WCHAR *szUrl;   /* Link URL string */
+            HRGN hRgn;      /* Region of the link */
+        } Link;
+        struct
+        {
+            UINT Dummy;
+        } Text;
+    } u;
+} DOC_ITEM, *PDOC_ITEM;
+
+typedef struct
+{
+    HWND      Self;         /* The window handle for this control */
+    PDOC_ITEM Items;        /* Address to the first document item */
+    BOOL      HasFocus;     /* Whether the control has the input focus */
+    int       MouseDownID;  /* ID of the link that the mouse button first selected */
+    HFONT     Font;         /* Handle to the font for text */
+    HFONT     LinkFont;     /* Handle to the font for links */
+    COLORREF  TextColor;    /* Color of the text */
+    COLORREF  LinkColor;    /* Color of links */
+    COLORREF  VisitedColor; /* Color of visited links */
+} SYSLINK_INFO;
+
+static const WCHAR SL_LINKOPEN[] =  { '<','a', 0 };
+static const WCHAR SL_HREF[] =      { 'h','r','e','f','=','\"',0 };
+static const WCHAR SL_ID[] =        { 'i','d','=','\"',0 };
+static const WCHAR SL_LINKCLOSE[] = { '<','/','a','>',0 };
+
+/* Control configuration constants */
+
+#define SL_LEFTMARGIN   (0)
+#define SL_TOPMARGIN    (0)
+#define SL_RIGHTMARGIN  (0)
+#define SL_BOTTOMMARGIN (0)
+
+/***********************************************************************
+ * SYSLINK_FreeDocItem
+ * Frees all data and gdi objects associated with a document item
+ */
+static VOID SYSLINK_FreeDocItem (PDOC_ITEM DocItem)
+{
+    if(DocItem->Type == slLink)
+    {
+        if(DocItem->u.Link.szID != NULL)
+        {
+            SYSLINK_Free(DocItem->u.Link.szID);
+        }
+        if(DocItem->u.Link.szUrl != NULL)
+        {
+            SYSLINK_Free(DocItem->u.Link.szUrl);
+        }
+    }
+
+    if(DocItem->Type == slLink && DocItem->u.Link.hRgn != NULL)
+    {
+        DeleteObject(DocItem->u.Link.hRgn);
+    }
+
+    /* we don't free Text because it's just a pointer to a character in the
+       entire window text string */
+
+    SYSLINK_Free(DocItem);
+}
+
+/***********************************************************************
+ * SYSLINK_AppendDocItem
+ * Create and append a new document item.
+ */
+static PDOC_ITEM SYSLINK_AppendDocItem (SYSLINK_INFO *infoPtr, LPWSTR Text, UINT textlen,
+                                        SL_ITEM_TYPE type, PDOC_ITEM LastItem)
+{
+    PDOC_ITEM Item;
+    Item = SYSLINK_Alloc(sizeof(DOC_ITEM) + ((textlen + 1) * sizeof(WCHAR)));
+    if(Item == NULL)
+    {
+        ERR("Failed to alloc DOC_ITEM structure!\n");
+        return NULL;
+    }
+    textlen = min(textlen, lstrlenW(Text));
+
+    Item->Next = NULL;
+    Item->Text = (LPWSTR)(Item + 1);
+    Item->nText = textlen;
+    Item->Type = type;
+    Item->Blocks = NULL;
+    
+    if(LastItem != NULL)
+    {
+        LastItem->Next = Item;
+    }
+    else
+    {
+        infoPtr->Items = Item;
+    }
+    
+    lstrcpynW(Item->Text, Text, textlen + 1);
+    Item->Text[textlen] = 0;
+    
+    return Item;
+}
+
+/***********************************************************************
+ * SYSLINK_ClearDoc
+ * Clears the document tree
+ */
+static VOID SYSLINK_ClearDoc (SYSLINK_INFO *infoPtr)
+{
+    PDOC_ITEM Item, Next;
+    
+    Item = infoPtr->Items;
+    while(Item != NULL)
+    {
+        Next = Item->Next;
+        SYSLINK_FreeDocItem(Item);
+        Item = Next;
+    }
+    
+    infoPtr->Items = NULL;
+}
+
+/***********************************************************************
+ * SYSLINK_ParseText
+ * Parses the window text string and creates a document. Returns the
+ * number of document items created.
+ */
+static UINT SYSLINK_ParseText (SYSLINK_INFO *infoPtr, LPWSTR Text)
+{
+    WCHAR *current, *textstart, *linktext, *firsttag;
+    int taglen = 0, textlen, linklen, docitems = 0;
+    PDOC_ITEM Last = NULL;
+    SL_ITEM_TYPE CurrentType = slText;
+    DWORD Style;
+    LPWSTR lpID, lpUrl;
+    UINT lenId, lenUrl;
+
+    Style = GetWindowLongW(infoPtr->Self, GWL_STYLE);
+
+    firsttag = NULL;
+    textstart = NULL;
+    linktext = NULL;
+    textlen = 0;
+    linklen = 0;
+    
+    for(current = (WCHAR*)Text; *current != 0;)
+    {
+        if(*current == '<')
+        {
+            if(!StrCmpNIW(current, SL_LINKOPEN, 2) && (CurrentType == slText))
+            {
+                BOOL ValidParam = FALSE, ValidLink = FALSE;
+
+                switch (*(current + 2))
+                {
+                case '>':
+                    /* we just have to deal with a <a> tag */
+                    taglen = 3;
+                    ValidLink = TRUE;
+                    ValidParam = TRUE;
+                    firsttag = current;
+                    linklen = 0;
+                    lpID = NULL;
+                    lpUrl = NULL;
+                    break;
+                case ' ':
+                {
+                    /* we expect parameters, parse them */
+                    LPWSTR *CurrentParameter = NULL;
+                    UINT *CurrentParameterLen = NULL;
+                    WCHAR *tmp;
+
+                    taglen = 3;
+                    tmp = current + taglen;
+                    lpID = NULL;
+                    lpUrl = NULL;
+                    
+CheckParameter:
+                    /* compare the current position with all known parameters */
+                    if(!StrCmpNIW(tmp, SL_HREF, 6))
+                    {
+                        taglen += 6;
+                        ValidParam = TRUE;
+                        CurrentParameter = &lpUrl;
+                        CurrentParameterLen = &lenUrl;
+                    }
+                    else if(!StrCmpNIW(tmp, SL_ID, 4))
+                    {
+                        taglen += 4;
+                        ValidParam = TRUE;
+                        CurrentParameter = &lpID;
+                        CurrentParameterLen = &lenId;
+                    }
+                    else
+                    {
+                        ValidParam = FALSE;
+                    }
+                    
+                    if(ValidParam)
+                    {
+                        /* we got a known parameter, now search until the next " character.
+                           If we can't find a " character, there's a syntax error and we just assume it's text */
+                        ValidParam = FALSE;
+                        *CurrentParameter = current + taglen;
+                        *CurrentParameterLen = 0;
+
+                        for(tmp = *CurrentParameter; *tmp != 0; tmp++)
+                        {
+                            taglen++;
+                            if(*tmp == '\"')
+                            {
+                                ValidParam = TRUE;
+                                tmp++;
+                                break;
+                            }
+                            (*CurrentParameterLen)++;
+                        }
+                    }
+                    if(ValidParam)
+                    {
+                        /* we're done with this parameter, now there are only 2 possibilities:
+                         * 1. another parameter is coming, so expect a ' ' (space) character
+                         * 2. the tag is being closed, so expect a '<' character
+                         */
+                        switch(*tmp)
+                        {
+                        case ' ':
+                            /* we expect another parameter, do the whole thing again */
+                            taglen++;
+                            tmp++;
+                            goto CheckParameter;
+
+                        case '>':
+                            /* the tag is being closed, we're done */
+                            ValidLink = TRUE;
+                            taglen++;
+                            break;
+                        default:
+                            tmp++;
+                            break;
+                        }
+                    }
+                    
+                    break;
+                }
+                }
+                
+                if(ValidLink && ValidParam)
+                {
+                    /* the <a ...> tag appears to be valid. save all information
+                       so we can add the link if we find a valid </a> tag later */
+                    CurrentType = slLink;
+                    linktext = current + taglen;
+                    linklen = 0;
+                    firsttag = current;
+                }
+                else
+                {
+                    taglen = 1;
+                    lpID = NULL;
+                    lpUrl = NULL;
+                    if(textstart == NULL)
+                    {
+                        textstart = current;
+                    }
+                }
+            }
+            else if(!StrCmpNIW(current, SL_LINKCLOSE, 4) && (CurrentType == slLink) && firsttag)
+            {
+                /* there's a <a...> tag opened, first add the previous text, if present */
+                if(textstart != NULL && textlen > 0 && firsttag > textstart)
+                {
+                    Last = SYSLINK_AppendDocItem(infoPtr, textstart, firsttag - textstart, slText, Last);
+                    if(Last == NULL)
+                    {
+                        ERR("Unable to create new document item!\n");
+                        return docitems;
+                    }
+                    docitems++;
+                    textstart = NULL;
+                    textlen = 0;
+                }
+                
+                /* now it's time to add the link to the document */
+                current += 4;
+                if(linktext != NULL && linklen > 0)
+                {
+                    Last = SYSLINK_AppendDocItem(infoPtr, linktext, linklen, slLink, Last);
+                    if(Last == NULL)
+                    {
+                        ERR("Unable to create new document item!\n");
+                        return docitems;
+                    }
+                    docitems++;
+                    if(CurrentType == slLink)
+                    {
+                        int nc;
+
+                        if(!(Style & WS_DISABLED))
+                        {
+                            Last->u.Link.state |= LIS_ENABLED;
+                        }
+                        /* Copy the tag parameters */
+                        if(lpID != NULL)
+                        {
+                            nc = min(lenId, lstrlenW(lpID));
+                            nc = min(nc, MAX_LINKID_TEXT);
+                            Last->u.Link.szID = SYSLINK_Alloc((MAX_LINKID_TEXT + 1) * sizeof(WCHAR));
+                            if(Last->u.Link.szID != NULL)
+                            {
+                                lstrcpynW(Last->u.Link.szID, lpID, nc + 1);
+                                Last->u.Link.szID[nc] = 0;
+                            }
+                        }
+                        else
+                            Last->u.Link.szID = NULL;
+                        if(lpUrl != NULL)
+                        {
+                            nc = min(lenUrl, lstrlenW(lpUrl));
+                            nc = min(nc, L_MAX_URL_LENGTH);
+                            Last->u.Link.szUrl = SYSLINK_Alloc((L_MAX_URL_LENGTH + 1) * sizeof(WCHAR));
+                            if(Last->u.Link.szUrl != NULL)
+                            {
+                                lstrcpynW(Last->u.Link.szUrl, lpUrl, nc + 1);
+                                Last->u.Link.szUrl[nc] = 0;
+                            }
+                        }
+                        else
+                            Last->u.Link.szUrl = NULL;
+                    }
+                    linktext = NULL;
+                }
+                CurrentType = slText;
+                firsttag = NULL;
+                textstart = NULL;
+                continue;
+            }
+            else
+            {
+                /* we don't know what tag it is, so just continue */
+                taglen = 1;
+                linklen++;
+                if(CurrentType == slText && textstart == NULL)
+                {
+                    textstart = current;
+                }
+            }
+            
+            textlen += taglen;
+            current += taglen;
+        }
+        else
+        {
+            textlen++;
+            linklen++;
+
+            /* save the pointer of the current text item if we couldn't find a tag */
+            if(textstart == NULL && CurrentType == slText)
+            {
+                textstart = current;
+            }
+            
+            current++;
+        }
+    }
+    
+    if(textstart != NULL && textlen > 0)
+    {
+        Last = SYSLINK_AppendDocItem(infoPtr, textstart, textlen, CurrentType, Last);
+        if(Last == NULL)
+        {
+            ERR("Unable to create new document item!\n");
+            return docitems;
+        }
+        if(CurrentType == slLink)
+        {
+            int nc;
+
+            if(!(Style & WS_DISABLED))
+            {
+                Last->u.Link.state |= LIS_ENABLED;
+            }
+            /* Copy the tag parameters */
+            if(lpID != NULL)
+            {
+                nc = min(lenId, lstrlenW(lpID));
+                nc = min(nc, MAX_LINKID_TEXT);
+                Last->u.Link.szID = SYSLINK_Alloc((MAX_LINKID_TEXT + 1) * sizeof(WCHAR));
+                if(Last->u.Link.szID != NULL)
+                {
+                    lstrcpynW(Last->u.Link.szID, lpID, nc + 1);
+                    Last->u.Link.szID[nc] = 0;
+                }
+            }
+            else
+                Last->u.Link.szID = NULL;
+            if(lpUrl != NULL)
+            {
+                nc = min(lenUrl, lstrlenW(lpUrl));
+                nc = min(nc, L_MAX_URL_LENGTH);
+                Last->u.Link.szUrl = SYSLINK_Alloc((L_MAX_URL_LENGTH + 1) * sizeof(WCHAR));
+                if(Last->u.Link.szUrl != NULL)
+                {
+                    lstrcpynW(Last->u.Link.szUrl, lpUrl, nc + 1);
+                    Last->u.Link.szUrl[nc] = 0;
+                }
+            }
+            else
+                Last->u.Link.szUrl = NULL;
+        }
+        docitems++;
+    }
+
+    if(linktext != NULL && linklen > 0)
+    {
+        /* we got a unclosed link, just display the text */
+        Last = SYSLINK_AppendDocItem(infoPtr, linktext, linklen, slText, Last);
+        if(Last == NULL)
+        {
+            ERR("Unable to create new document item!\n");
+            return docitems;
+        }
+        docitems++;
+    }
+
+    return docitems;
+}
+
+/***********************************************************************
+ * SYSLINK_RepaintLink
+ * Repaints a link.
+ */
+static VOID SYSLINK_RepaintLink (SYSLINK_INFO *infoPtr, PDOC_ITEM DocItem)
+{
+    if(DocItem->Type != slLink)
+    {
+        ERR("DocItem not a link!\n");
+        return;
+    }
+    
+    if(DocItem->u.Link.hRgn != NULL)
+    {
+        /* repaint the region */
+        RedrawWindow(infoPtr->Self, NULL, DocItem->u.Link.hRgn, RDW_INVALIDATE | RDW_UPDATENOW);
+    }
+}
+
+/***********************************************************************
+ * SYSLINK_GetLinkItemByIndex
+ * Retreives a document link by it's index
+ */
+static PDOC_ITEM SYSLINK_GetLinkItemByIndex (SYSLINK_INFO *infoPtr, int iLink)
+{
+    PDOC_ITEM Current = infoPtr->Items;
+
+    while(Current != NULL)
+    {
+        if((Current->Type == slLink) && (iLink-- <= 0))
+        {
+            return Current;
+        }
+        Current = Current->Next;
+    }
+    return NULL;
+}
+
+/***********************************************************************
+ * SYSLINK_GetFocusLink
+ * Retreives the link that has the LIS_FOCUSED bit
+ */
+static PDOC_ITEM SYSLINK_GetFocusLink (SYSLINK_INFO *infoPtr, int *LinkId)
+{
+    PDOC_ITEM Current = infoPtr->Items;
+    int id = 0;
+
+    while(Current != NULL)
+    {
+        if((Current->Type == slLink))
+        {
+            if(Current->u.Link.state & LIS_FOCUSED)
+            {
+                if(LinkId != NULL)
+                    *LinkId = id;
+                return Current;
+            }
+            id++;
+        }
+        Current = Current->Next;
+    }
+    return NULL;
+}
+
+/***********************************************************************
+ * SYSLINK_GetNextLink
+ * Gets the next link
+ */
+static PDOC_ITEM SYSLINK_GetNextLink (SYSLINK_INFO *infoPtr, PDOC_ITEM Current)
+{
+    for(Current = (Current != NULL ? Current->Next : infoPtr->Items);
+        Current != NULL;
+        Current = Current->Next)
+    {
+        if(Current->Type == slLink)
+        {
+            return Current;
+        }
+    }
+    return NULL;
+}
+
+/***********************************************************************
+ * SYSLINK_GetPrevLink
+ * Gets the previous link
+ */
+static PDOC_ITEM SYSLINK_GetPrevLink (SYSLINK_INFO *infoPtr, PDOC_ITEM Current)
+{
+    if(Current == NULL)
+    {
+        /* returns the last link */
+        PDOC_ITEM Last = NULL;
+        
+        for(Current = infoPtr->Items; Current != NULL; Current = Current->Next)
+        {
+            if(Current->Type == slLink)
+            {
+                Last = Current;
+            }
+        }
+        return Last;
+    }
+    else
+    {
+        /* returns the previous link */
+        PDOC_ITEM Cur, Prev = NULL;
+        
+        for(Cur = infoPtr->Items; Cur != NULL; Cur = Cur->Next)
+        {
+            if(Cur == Current)
+            {
+                break;
+            }
+            if(Cur->Type == slLink)
+            {
+                Prev = Cur;
+            }
+        }
+        return Prev;
+    }
+}
+
+/***********************************************************************
+ * SYSLINK_WrapLine
+ * Tries to wrap a line.
+ */
+static BOOL SYSLINK_WrapLine (HDC hdc, LPWSTR Text, WCHAR BreakChar, int *LineLen, int nFit, LPSIZE Extent, int Width)
+{
+    WCHAR *Current;
+
+    if(nFit == *LineLen)
+    {
+        return FALSE;
+    }
+
+    *LineLen = nFit;
+
+    Current = Text + nFit;
+    
+    /* check if we're in the middle of a word */
+    if((*Current) != BreakChar)
+    {
+        /* search for the beginning of the word */
+        while(Current > Text && (*(Current - 1)) != BreakChar)
+        {
+            Current--;
+            (*LineLen)--;
+        }
+        
+        if((*LineLen) == 0)
+        {
+            Extent->cx = 0;
+            Extent->cy = 0;
+        }
+        return TRUE;
+    }
+
+    return TRUE;
+}
+
+/***********************************************************************
+ * SYSLINK_Render
+ * Renders the document in memory
+ */
+static VOID SYSLINK_Render (SYSLINK_INFO *infoPtr, HDC hdc)
+{
+    RECT rc;
+    PDOC_ITEM Current;
+    HGDIOBJ hOldFont;
+    int x, y, LineHeight;
+    TEXTMETRICW tm;
+    
+    GetClientRect(infoPtr->Self, &rc);
+    rc.right -= SL_RIGHTMARGIN;
+    rc.bottom -= SL_BOTTOMMARGIN;
+    
+    if(rc.right - SL_LEFTMARGIN < 0 || rc.bottom - SL_TOPMARGIN < 0) return;
+    
+    hOldFont = SelectObject(hdc, infoPtr->Font);
+    GetTextMetricsW(hdc, &tm);
+    
+    x = SL_LEFTMARGIN;
+    y = SL_TOPMARGIN;
+    LineHeight = 0;
+    
+    for(Current = infoPtr->Items; Current != NULL; Current = Current->Next)
+    {
+        int n, nBlocks;
+        LPWSTR tx;
+        PDOC_TEXTBLOCK bl, cbl;
+        INT nFit;
+        SIZE szDim;
+        
+        if(Current->nText == 0)
+        {
+            ERR("DOC_ITEM with no text?!\n");
+            continue;
+        }
+        
+        tx = Current->Text;
+        n = Current->nText;
+        bl = Current->Blocks;
+        nBlocks = 0;
+        
+        if(Current->Type == slText)
+        {
+            SelectObject(hdc, infoPtr->Font);
+        }
+        else if(Current->Type == slLink)
+        {
+            SelectObject(hdc, infoPtr->LinkFont);
+        }
+        
+        while(n > 0)
+        {
+            if(GetTextExtentExPointW(hdc, tx, n, rc.right - x, &nFit, NULL, &szDim))
+            {
+                int LineLen = n;
+                BOOL Wrap = SYSLINK_WrapLine(hdc, tx, tm.tmBreakChar, &LineLen, nFit, &szDim, rc.right - x);
+                
+                if(LineLen == 0)
+                {
+                    if(x > SL_LEFTMARGIN)
+                    {
+                        /* move one line down, the word didn't fit into the line */
+                        x = SL_LEFTMARGIN;
+                        y += LineHeight;
+                        LineHeight = 0;
+                        continue;
+                    }
+                    else
+                    {
+                        /* the word starts at the beginning of the line and doesn't
+                           fit into the line, so break it at the last character that fits */
+                        LineLen = max(nFit, 1);
+                    }
+                }
+                
+                if(LineLen != n)
+                {
+                    GetTextExtentExPointW(hdc, tx, LineLen, rc.right - x, NULL, NULL, &szDim);
+                }
+                
+                if(bl != NULL)
+                {
+                    bl = SYSLINK_ReAlloc(bl, ++nBlocks * sizeof(DOC_TEXTBLOCK));
+                }
+                else
+                {
+                    bl = SYSLINK_Alloc(++nBlocks * sizeof(DOC_TEXTBLOCK));
+                }
+                
+                if(bl != NULL)
+                {
+                    cbl = bl + nBlocks - 1;
+                    
+                    cbl->nChars = LineLen;
+                    cbl->rc.left = x;
+                    cbl->rc.top = y;
+                    cbl->rc.right = x + szDim.cx;
+                    cbl->rc.bottom = y + szDim.cy;
+                    
+                    x += szDim.cx;
+                    LineHeight = max(LineHeight, szDim.cy);
+                    
+                    /* (re)calculate the link's region */
+                    if(Current->Type == slLink)
+                    {
+                        if(nBlocks <= 1)
+                        {
+                            if(Current->u.Link.hRgn != NULL)
+                            {
+                                DeleteObject(Current->u.Link.hRgn);
+                            }
+                            /* initialize the link's hRgn */
+                            Current->u.Link.hRgn = CreateRectRgnIndirect(&cbl->rc);
+                        }
+                        else if(Current->u.Link.hRgn != NULL)
+                        {
+                            HRGN hrgn;
+                            hrgn = CreateRectRgnIndirect(&cbl->rc);
+                            /* add the rectangle */
+                            CombineRgn(Current->u.Link.hRgn, Current->u.Link.hRgn, hrgn, RGN_OR);
+                            DeleteObject(hrgn);
+                        }
+                    }
+                    
+                    if(Wrap)
+                    {
+                        x = SL_LEFTMARGIN;
+                        y += LineHeight;
+                        LineHeight = 0;
+                    }
+                }
+                else
+                {
+                    ERR("Failed to alloc DOC_TEXTBLOCK structure!\n");
+                    break;
+                }
+                n -= LineLen;
+                tx += LineLen;
+            }
+            else
+            {
+                ERR("GetTextExtentExPoint() failed?!\n");
+                n--;
+            }
+        }
+        Current->Blocks = bl;
+    }
+    
+    SelectObject(hdc, hOldFont);
+}
+
+/***********************************************************************
+ * SYSLINK_Draw
+ * Draws the SysLink control.
+ */
+static LRESULT SYSLINK_Draw (SYSLINK_INFO *infoPtr, HDC hdc)
+{
+    RECT rc;
+    PDOC_ITEM Current;
+    HFONT hOldFont;
+    COLORREF OldTextColor, OldBkColor;
+
+    hOldFont = SelectObject(hdc, infoPtr->Font);
+    OldTextColor = SetTextColor(hdc, infoPtr->TextColor);
+    OldBkColor = SetBkColor(hdc, GetSysColor(COLOR_BTNFACE));
+    
+    GetClientRect(infoPtr->Self, &rc);
+    rc.right -= SL_RIGHTMARGIN + SL_LEFTMARGIN;
+    rc.bottom -= SL_BOTTOMMARGIN + SL_TOPMARGIN;
+
+    if(rc.right < 0 || rc.bottom < 0) return 0;
+
+    for(Current = infoPtr->Items; Current != NULL; Current = Current->Next)
+    {
+        int n;
+        LPWSTR tx;
+        PDOC_TEXTBLOCK bl;
+        
+        bl = Current->Blocks;
+        if(bl != NULL)
+        {
+            tx = Current->Text;
+            n = Current->nText;
+
+            if(Current->Type == slText)
+            {
+                 SelectObject(hdc, infoPtr->Font);
+                 SetTextColor(hdc, infoPtr->TextColor);
+            }
+            else
+            {
+                 SelectObject(hdc, infoPtr->LinkFont);
+                 SetTextColor(hdc, (!(Current->u.Link.state & LIS_VISITED) ? infoPtr->LinkColor : infoPtr->VisitedColor));
+            }
+
+            while(n > 0)
+            {
+                ExtTextOutW(hdc, bl->rc.left, bl->rc.top, ETO_OPAQUE | ETO_CLIPPED, &bl->rc, tx, bl->nChars, NULL);
+                if((Current->Type == slLink) && (Current->u.Link.state & LIS_FOCUSED) && infoPtr->HasFocus)
+                {
+                    COLORREF PrevColor;
+                    PrevColor = SetBkColor(hdc, OldBkColor);
+                    DrawFocusRect(hdc, &bl->rc);
+                    SetBkColor(hdc, PrevColor);
+                }
+                tx += bl->nChars;
+                n -= bl->nChars;
+                bl++;
+            }
+        }
+    }
+
+    SetBkColor(hdc, OldBkColor);
+    SetTextColor(hdc, OldTextColor);
+    SelectObject(hdc, hOldFont);
+    
+    return 0;
+}
+
+
+/***********************************************************************
+ * SYSLINK_Paint
+ * Handles the WM_PAINT message.
+ */
+static LRESULT SYSLINK_Paint (SYSLINK_INFO *infoPtr)
+{
+    HDC hdc;
+    PAINTSTRUCT ps;
+    hdc = BeginPaint (infoPtr->Self, &ps);
+    SYSLINK_Draw (infoPtr, hdc);
+    EndPaint (infoPtr->Self, &ps);
+    return 0;
+}
+
+
+/***********************************************************************
+ *           SYSLINK_SetFont
+ * Set new Font for the SysLink control.
+ */
+static HFONT SYSLINK_SetFont (SYSLINK_INFO *infoPtr, HFONT hFont, BOOL bRedraw)
+{
+    HDC hdc;
+    LOGFONTW lf;
+    HFONT hOldFont = infoPtr->Font;
+    infoPtr->Font = hFont;
+    
+    /* free the underline font */
+    if(infoPtr->LinkFont != NULL)
+    {
+        DeleteObject(infoPtr->LinkFont);
+        infoPtr->LinkFont = NULL;
+    }
+
+    /* Render text position and word wrapping in memory */
+    hdc = GetDC(infoPtr->Self);
+    if(hdc != NULL)
+    {
+        /* create a new underline font */
+        if(GetObjectW(infoPtr->Font, sizeof(LOGFONTW), &lf))
+        {
+            lf.lfUnderline = TRUE;
+            infoPtr->LinkFont = CreateFontIndirectW(&lf);
+        }
+        else
+        {
+            ERR("Failed to create link font!\n");
+        }
+
+        SYSLINK_Render(infoPtr, hdc);
+        ReleaseDC(infoPtr->Self, hdc);
+    }
+    
+    if(bRedraw)
+    {
+        RedrawWindow(infoPtr->Self, NULL, NULL, RDW_INVALIDATE | RDW_UPDATENOW);
+    }
+    
+    return hOldFont;
+}
+
+/***********************************************************************
+ *           SYSLINK_SetText
+ * Set new text for the SysLink control.
+ */
+static LRESULT SYSLINK_SetText (SYSLINK_INFO *infoPtr, LPWSTR Text)
+{
+    int textlen;
+
+    /* clear the document */
+    SYSLINK_ClearDoc(infoPtr);
+    
+    textlen = lstrlenW(Text);
+    if(Text == NULL || textlen == 0)
+    {
+        return TRUE;
+    }
+    
+    /* let's parse the string and create a document */
+    if(SYSLINK_ParseText(infoPtr, Text) > 0)
+    {
+        /* Render text position and word wrapping in memory */
+        HDC hdc = GetDC(infoPtr->Self);
+        SYSLINK_Render(infoPtr, hdc);
+        SYSLINK_Draw(infoPtr, hdc);
+        ReleaseDC(infoPtr->Self, hdc);
+    }
+    
+    return TRUE;
+}
+
+/***********************************************************************
+ *           SYSLINK_SetFocusLink
+ * Updates the focus status bits and focusses the specified link.
+ * If no document item is specified, the focus bit will be removed from all links.
+ * Returns the previous focused item.
+ */
+static PDOC_ITEM SYSLINK_SetFocusLink (SYSLINK_INFO *infoPtr, PDOC_ITEM DocItem)
+{
+    PDOC_ITEM Current, PrevFocus = NULL;
+    
+    for(Current = infoPtr->Items; Current != NULL; Current = Current->Next)
+    {
+        if(Current->Type == slLink)
+        {
+            if((PrevFocus == NULL) && (Current->u.Link.state & LIS_FOCUSED))
+            {
+                PrevFocus = Current;
+            }
+            
+            if(Current == DocItem)
+            {
+                Current->u.Link.state |= LIS_FOCUSED;
+            }
+            else
+            {
+                Current->u.Link.state &= ~LIS_FOCUSED;
+            }
+        }
+    }
+    
+    return PrevFocus;
+}
+
+/***********************************************************************
+ *           SYSLINK_SetItem
+ * Sets the states and attributes of a link item.
+ */
+static LRESULT SYSLINK_SetItem (SYSLINK_INFO *infoPtr, PLITEM Item)
+{
+    PDOC_ITEM di;
+    BOOL Repaint = FALSE;
+    BOOL Ret = TRUE;
+
+    if(!(Item->mask & LIF_ITEMINDEX) || !(Item->mask & (LIF_FLAGSMASK)))
+    {
+        ERR("Invalid Flags!\n");
+        return FALSE;
+    }
+
+    di = SYSLINK_GetLinkItemByIndex(infoPtr, Item->iLink);
+    if(di == NULL)
+    {
+        ERR("Link %d couldn't be found\n", Item->iLink);
+        return FALSE;
+    }
+    
+    if(Item->mask & LIF_STATE)
+    {
+        UINT oldstate = di->u.Link.state;
+        /* clear the masked bits */
+        di->u.Link.state &= ~(Item->stateMask & LIS_MASK);
+        /* copy the bits */
+        di->u.Link.state |= (Item->state & Item->stateMask) & LIS_MASK;
+        Repaint = (oldstate != di->u.Link.state);
+        
+        /* update the focus */
+        SYSLINK_SetFocusLink(infoPtr, ((di->u.Link.state & LIS_FOCUSED) ? di : NULL));
+    }
+
+    if(Item->mask & LIF_ITEMID)
+    {
+        if(!di->u.Link.szID)
+        {
+            di->u.Link.szID = SYSLINK_Alloc((MAX_LINKID_TEXT + 1) * sizeof(WCHAR));
+            if(!Item->szID)
+            {
+                ERR("Unable to allocate memory for link id\n");
+                Ret = FALSE;
+            }
+        }
+        if(di->u.Link.szID)
+        {
+            lstrcpynW(di->u.Link.szID, Item->szID, MAX_LINKID_TEXT + 1);
+        }
+    }
+
+    if(Item->mask & LIF_URL)
+    {
+        if(!di->u.Link.szUrl)
+        {
+            di->u.Link.szUrl = SYSLINK_Alloc((MAX_LINKID_TEXT + 1) * sizeof(WCHAR));
+            if(!Item->szUrl)
+            {
+                ERR("Unable to allocate memory for link url\n");
+                Ret = FALSE;
+            }
+        }
+        if(di->u.Link.szUrl)
+        {
+            lstrcpynW(di->u.Link.szUrl, Item->szUrl, MAX_LINKID_TEXT + 1);
+        }
+    }
+    
+    if(Repaint)
+    {
+        SYSLINK_RepaintLink(infoPtr, di);
+    }
+    
+    return Ret;
+}
+
+/***********************************************************************
+ *           SYSLINK_GetItem
+ * Retrieves the states and attributes of a link item.
+ */
+static LRESULT SYSLINK_GetItem (SYSLINK_INFO *infoPtr, PLITEM Item)
+{
+    PDOC_ITEM di;
+    
+    if(!(Item->mask & LIF_ITEMINDEX) || !(Item->mask & (LIF_FLAGSMASK)))
+    {
+        ERR("Invalid Flags!\n");
+        return FALSE;
+    }
+    
+    di = SYSLINK_GetLinkItemByIndex(infoPtr, Item->iLink);
+    if(di == NULL)
+    {
+        ERR("Link %d couldn't be found\n", Item->iLink);
+        return FALSE;
+    }
+    
+    if(Item->mask & LIF_STATE)
+    {
+        Item->state = (di->u.Link.state & Item->stateMask);
+        if(!infoPtr->HasFocus)
+        {
+            /* remove the LIS_FOCUSED bit if the control doesn't have focus */
+            Item->state &= ~LIS_FOCUSED;
+        }
+    }
+    
+    if(Item->mask & LIF_ITEMID)
+    {
+        if(di->u.Link.szID)
+        {
+            lstrcpynW(Item->szID, di->u.Link.szID, MAX_LINKID_TEXT + 1);
+        }
+        else
+        {
+            Item->szID[0] = 0;
+        }
+    }
+    
+    if(Item->mask & LIF_URL)
+    {
+        if(di->u.Link.szUrl)
+        {
+            lstrcpynW(Item->szUrl, di->u.Link.szUrl, L_MAX_URL_LENGTH + 1);
+        }
+        else
+        {
+            Item->szUrl[0] = 0;
+        }
+    }
+    
+    return TRUE;
+}
+
+/***********************************************************************
+ *           SYSLINK_HitTest
+ * Determines the link the user clicked on.
+ */
+static LRESULT SYSLINK_HitTest (SYSLINK_INFO *infoPtr, PLHITTESTINFO HitTest)
+{
+    PDOC_ITEM Current;
+    int id = 0;
+
+    for(Current = infoPtr->Items; Current != NULL; Current = Current->Next)
+    {
+        if(Current->Type == slLink)
+        {
+            if((Current->u.Link.hRgn != NULL) &&
+               PtInRegion(Current->u.Link.hRgn, HitTest->pt.x, HitTest->pt.y))
+            {
+                HitTest->item.mask = 0;
+                HitTest->item.iLink = id;
+                HitTest->item.state = 0;
+                HitTest->item.stateMask = 0;
+                if(Current->u.Link.szID)
+                {
+                    lstrcpynW(HitTest->item.szID, Current->u.Link.szID, MAX_LINKID_TEXT + 1);
+                }
+                else
+                {
+                    HitTest->item.szID[0] = 0;
+                }
+                if(Current->u.Link.szUrl)
+                {
+                    lstrcpynW(HitTest->item.szUrl, Current->u.Link.szUrl, L_MAX_URL_LENGTH + 1);
+                }
+                else
+                {
+                    HitTest->item.szUrl[0] = 0;
+                }
+                return TRUE;
+            }
+            id++;
+        }
+    }
+    
+    return FALSE;
+}
+
+/***********************************************************************
+ *           SYSLINK_GetIdealHeight
+ * Returns the preferred height of a link at the current control's width.
+ */
+static LRESULT SYSLINK_GetIdealHeight (SYSLINK_INFO *infoPtr)
+{
+    HDC hdc = GetDC(infoPtr->Self);
+    if(hdc != NULL)
+    {
+        LRESULT height;
+        TEXTMETRICW tm;
+        HGDIOBJ hOldFont = SelectObject(hdc, infoPtr->Font);
+        
+        if(GetTextMetricsW(hdc, &tm))
+        {
+            height = tm.tmHeight;
+        }
+        else
+        {
+            height = 0;
+        }
+        SelectObject(hdc, hOldFont);
+        ReleaseDC(infoPtr->Self, hdc);
+        
+        return height;
+    }
+    return 0;
+}
+
+/***********************************************************************
+ *           SYSLINK_SendParentNotify
+ * Sends a WM_NOTIFY message to the parent window.
+ */
+static LRESULT SYSLINK_SendParentNotify (SYSLINK_INFO *infoPtr, UINT code, PDOC_ITEM Link, int iLink)
+{
+    NMLINK nml;
+
+    nml.hdr.hwndFrom = infoPtr->Self;
+    nml.hdr.idFrom = GetWindowLongW(infoPtr->Self, GWL_ID);
+    nml.hdr.code = code;
+
+    nml.item.mask = 0;
+    nml.item.iLink = iLink;
+    nml.item.state = 0;
+    nml.item.stateMask = 0;
+    if(Link->u.Link.szID)
+    {
+        lstrcpynW(nml.item.szID, Link->u.Link.szID, MAX_LINKID_TEXT + 1);
+    }
+    else
+    {
+        nml.item.szID[0] = 0;
+    }
+    if(Link->u.Link.szUrl)
+    {
+        lstrcpynW(nml.item.szUrl, Link->u.Link.szUrl, L_MAX_URL_LENGTH + 1);
+    }
+    else
+    {
+        nml.item.szUrl[0] = 0;
+    }
+
+    return SendMessageW(GetParent(infoPtr->Self), WM_NOTIFY, (WPARAM)nml.hdr.idFrom, (LPARAM)&nml);
+}
+
+/***********************************************************************
+ *           SYSLINK_SetFocus
+ * Handles receiving the input focus.
+ */
+static LRESULT SYSLINK_SetFocus (SYSLINK_INFO *infoPtr, HWND PrevFocusWindow)
+{
+    PDOC_ITEM Focus;
+    
+    infoPtr->HasFocus = TRUE;
+
+#if 1
+    /* FIXME - How to detect whether SHIFT+TAB or just TAB has been pressed?
+     *         The problem is we could get this message without keyboard input, too
+     */
+    Focus = SYSLINK_GetFocusLink(infoPtr, NULL);
+    
+    if(Focus == NULL && (Focus = SYSLINK_GetNextLink(infoPtr, NULL)))
+    {
+        SYSLINK_SetFocusLink(infoPtr, Focus);
+    }
+#else
+    /* This is a temporary hack since I'm not really sure how to detect which link to select.
+       See message above! */
+    Focus = SYSLINK_GetNextLink(infoPtr, NULL);
+    if(Focus != NULL)
+    {
+        SYSLINK_SetFocusLink(infoPtr, Focus);
+    }
+#endif
+    
+    SYSLINK_RepaintLink(infoPtr, Focus);
+    
+    return 0;
+}
+
+/***********************************************************************
+ *           SYSLINK_KillFocus
+ * Handles losing the input focus.
+ */
+static LRESULT SYSLINK_KillFocus (SYSLINK_INFO *infoPtr, HWND NewFocusWindow)
+{
+    PDOC_ITEM Focus;
+    
+    infoPtr->HasFocus = FALSE;
+    Focus = SYSLINK_GetFocusLink(infoPtr, NULL);
+    
+    if(Focus != NULL)
+    {
+        SYSLINK_RepaintLink(infoPtr, Focus);
+    }
+
+    return 0;
+}
+
+/***********************************************************************
+ *           SYSLINK_LinkAtPt
+ * Returns a link at the specified position
+ */
+static PDOC_ITEM SYSLINK_LinkAtPt (SYSLINK_INFO *infoPtr, POINT *pt, int *LinkId, BOOL MustBeEnabled)
+{
+    PDOC_ITEM Current;
+    int id = 0;
+
+    for(Current = infoPtr->Items; Current != NULL; Current = Current->Next)
+    {
+        if((Current->Type == slLink) && (Current->u.Link.hRgn != NULL) &&
+           PtInRegion(Current->u.Link.hRgn, pt->x, pt->y) &&
+           (!MustBeEnabled || (MustBeEnabled && (Current->u.Link.state & LIS_ENABLED))))
+        {
+            if(LinkId != NULL)
+            {
+                *LinkId = id;
+            }
+            return Current;
+        }
+        id++;
+    }
+
+    return NULL;
+}
+
+/***********************************************************************
+ *           SYSLINK_LButtonDown
+ * Handles mouse clicks
+ */
+static LRESULT SYSLINK_LButtonDown (SYSLINK_INFO *infoPtr, DWORD Buttons, POINT *pt)
+{
+    PDOC_ITEM Current, Old;
+    int id;
+
+    Current = SYSLINK_LinkAtPt(infoPtr, pt, &id, TRUE);
+    if(Current != NULL)
+    {
+      Old = SYSLINK_SetFocusLink(infoPtr, Current);
+      if(Old != NULL && Old != Current)
+      {
+          SYSLINK_RepaintLink(infoPtr, Old);
+      }
+      infoPtr->MouseDownID = id;
+      SYSLINK_RepaintLink(infoPtr, Current);
+      SetFocus(infoPtr->Self);
+    }
+
+    return 0;
+}
+
+/***********************************************************************
+ *           SYSLINK_LButtonUp
+ * Handles mouse clicks
+ */
+static LRESULT SYSLINK_LButtonUp (SYSLINK_INFO *infoPtr, DWORD Buttons, POINT *pt)
+{
+    if(infoPtr->MouseDownID > -1)
+    {
+        PDOC_ITEM Current;
+        int id;
+        
+        Current = SYSLINK_LinkAtPt(infoPtr, pt, &id, TRUE);
+        if((Current != NULL) && (Current->u.Link.state & LIS_FOCUSED) && (infoPtr->MouseDownID == id))
+        {
+            SYSLINK_SendParentNotify(infoPtr, NM_CLICK, Current, id);
+        }
+    }
+
+    infoPtr->MouseDownID = -1;
+
+    return 0;
+}
+
+/***********************************************************************
+ *           SYSLINK_OnEnter
+ * Handles ENTER key events
+ */
+static BOOL SYSLINK_OnEnter (SYSLINK_INFO *infoPtr)
+{
+    if(infoPtr->HasFocus)
+    {
+        PDOC_ITEM Focus;
+        int id;
+        
+        Focus = SYSLINK_GetFocusLink(infoPtr, &id);
+        if(Focus != NULL)
+        {
+            SYSLINK_SendParentNotify(infoPtr, NM_RETURN, Focus, id);
+            return TRUE;
+        }
+    }
+    return FALSE;
+}
+
+/***********************************************************************
+ *           SYSKEY_SelectNextPrevLink
+ * Changes the currently focused link
+ */
+static BOOL SYSKEY_SelectNextPrevLink (SYSLINK_INFO *infoPtr, BOOL Prev)
+{
+    if(infoPtr->HasFocus)
+    {
+        PDOC_ITEM Focus;
+        int id;
+
+        Focus = SYSLINK_GetFocusLink(infoPtr, &id);
+        if(Focus != NULL)
+        {
+            PDOC_ITEM NewFocus, OldFocus;
+
+            if(Prev)
+                NewFocus = SYSLINK_GetPrevLink(infoPtr, Focus);
+            else
+                NewFocus = SYSLINK_GetNextLink(infoPtr, Focus);
+
+            if(NewFocus != NULL)
+            {
+                OldFocus = SYSLINK_SetFocusLink(infoPtr, NewFocus);
+
+                if(OldFocus != NewFocus)
+                {
+                    SYSLINK_RepaintLink(infoPtr, OldFocus);
+                }
+                SYSLINK_RepaintLink(infoPtr, NewFocus);
+                return TRUE;
+            }
+        }
+    }
+    return FALSE;
+}
+
+/***********************************************************************
+ *           SYSKEY_SelectNextPrevLink
+ * Determines if there's a next or previous link to decide whether the control
+ * should capture the tab key message
+ */
+static BOOL SYSLINK_NoNextLink (SYSLINK_INFO *infoPtr, BOOL Prev)
+{
+    PDOC_ITEM Focus, NewFocus;
+
+    Focus = SYSLINK_GetFocusLink(infoPtr, NULL);
+    if(Prev)
+        NewFocus = SYSLINK_GetPrevLink(infoPtr, Focus);
+    else
+        NewFocus = SYSLINK_GetNextLink(infoPtr, Focus);
+
+    return NewFocus == NULL;
+}
+
+/***********************************************************************
+ *           SysLinkWindowProc
+ */
+static LRESULT WINAPI SysLinkWindowProc(HWND hwnd, UINT message,
+                                        WPARAM wParam, LPARAM lParam)
+{
+    SYSLINK_INFO *infoPtr;
+
+    TRACE("hwnd=%p msg=%04x wparam=%x lParam=%lx\n", hwnd, message, wParam, lParam);
+
+    infoPtr = (SYSLINK_INFO *)GetWindowLongW(hwnd, 0);
+
+    if (!infoPtr && message != WM_CREATE)
+        return DefWindowProcW( hwnd, message, wParam, lParam );
+
+    switch(message) {
+    case WM_PAINT:
+        return SYSLINK_Paint (infoPtr);
+
+    case WM_SETCURSOR:
+    {
+        LHITTESTINFO ht;
+        POINTS pt;
+        DWORD mp = GetMessagePos();
+        
+        pt = MAKEPOINTS(mp);
+        ht.pt.x = pt.x;
+        ht.pt.y = pt.y;
+        
+        ScreenToClient(infoPtr->Self, &ht.pt);
+        if(SYSLINK_HitTest (infoPtr, &ht))
+        {
+            SetCursor(LoadCursorW(0, (LPCWSTR)IDC_HAND));
+            return TRUE;
+        }
+        /* let the default window proc handle this message */
+        return DefWindowProcW(hwnd, message, wParam, lParam);
+
+    }
+
+    case WM_SIZE:
+    {
+        HDC hdc = GetDC(infoPtr->Self);
+        if(hdc != NULL)
+        {
+          SYSLINK_Render(infoPtr, hdc);
+          ReleaseDC(infoPtr->Self, hdc);
+        }
+        return 0;
+    }
+
+    case WM_GETFONT:
+        return (LRESULT)infoPtr->Font;
+
+    case WM_SETFONT:
+        return (LRESULT)SYSLINK_SetFont(infoPtr, (HFONT)wParam, (BOOL)lParam);
+
+    case WM_SETTEXT:
+        SYSLINK_SetText(infoPtr, (LPWSTR)lParam);
+        return DefWindowProcW(hwnd, message, wParam, lParam);
+
+    case WM_LBUTTONDOWN:
+    {
+        POINT pt;
+        pt.x = LOWORD(lParam);
+        pt.y = HIWORD(lParam);
+        return SYSLINK_LButtonDown(infoPtr, wParam, &pt);
+    }
+    case WM_LBUTTONUP:
+    {
+        POINT pt;
+        pt.x = LOWORD(lParam);
+        pt.y = HIWORD(lParam);
+        return SYSLINK_LButtonUp(infoPtr, wParam, &pt);
+    }
+    
+    case WM_KEYDOWN:
+    {
+        switch(wParam)
+        {
+        case VK_RETURN:
+            SYSLINK_OnEnter(infoPtr);
+            return 0;
+        case VK_TAB:
+        {
+            BOOL shift = GetKeyState(VK_SHIFT) & 0x8000;
+            SYSKEY_SelectNextPrevLink(infoPtr, shift);
+            return 0;
+        }
+        }
+        return DefWindowProcW(hwnd, message, wParam, lParam);
+    }
+    
+    case WM_GETDLGCODE:
+    {
+        LRESULT Ret = DLGC_HASSETSEL;
+        int vk = (lParam != 0 ? (int)((LPMSG)lParam)->wParam : 0);
+        switch(vk)
+        {
+        case VK_RETURN:
+            Ret |= DLGC_WANTMESSAGE;
+            break;
+        case VK_TAB:
+        {
+            BOOL shift = GetKeyState(VK_SHIFT) & 0x8000;
+            if(!SYSLINK_NoNextLink(infoPtr, shift))
+            {
+                Ret |= DLGC_WANTTAB;
+            }
+            else
+            {
+                Ret |= DLGC_WANTCHARS;
+            }
+            break;
+        }
+        }
+        return Ret;
+    }
+    
+    case WM_NCHITTEST:
+    {
+        POINT pt;
+        RECT rc;
+        pt.x = LOWORD(lParam);
+        pt.y = HIWORD(lParam);
+        
+        GetClientRect(infoPtr->Self, &rc);
+        ScreenToClient(infoPtr->Self, &pt);
+        if(pt.x < 0 || pt.y < 0 || pt.x > rc.right || pt.y > rc.bottom)
+        {
+            return HTNOWHERE;
+        }
+
+        if(SYSLINK_LinkAtPt(infoPtr, &pt, NULL, FALSE))
+        {
+            return HTCLIENT;
+        }
+        
+        return HTTRANSPARENT;
+    }
+
+    case LM_HITTEST:
+        return SYSLINK_HitTest(infoPtr, (PLHITTESTINFO)lParam);
+
+    case LM_SETITEM:
+        return SYSLINK_SetItem(infoPtr, (PLITEM)lParam);
+
+    case LM_GETITEM:
+        return SYSLINK_GetItem(infoPtr, (PLITEM)lParam);
+
+    case LM_GETIDEALHEIGHT:
+        return SYSLINK_GetIdealHeight(infoPtr);
+
+    case WM_SETFOCUS:
+        return SYSLINK_SetFocus(infoPtr, (HWND)wParam);
+
+    case WM_KILLFOCUS:
+        return SYSLINK_KillFocus(infoPtr, (HWND)wParam);
+
+    case WM_CREATE:
+        /* allocate memory for info struct */
+        infoPtr = (SYSLINK_INFO *)SYSLINK_Alloc (sizeof(SYSLINK_INFO));
+        if (!infoPtr) return -1;
+        SetWindowLongW (hwnd, 0, (DWORD)infoPtr);
+
+        /* initialize the info struct */
+        infoPtr->Self = hwnd;
+        infoPtr->Font = 0;
+        infoPtr->LinkFont = 0;
+        infoPtr->Items = NULL;
+        infoPtr->HasFocus = FALSE;
+        infoPtr->MouseDownID = -1;
+        infoPtr->TextColor = GetSysColor(COLOR_WINDOWTEXT);
+        infoPtr->LinkColor = GetSysColor(COLOR_HIGHLIGHT);
+        infoPtr->VisitedColor = GetSysColor(COLOR_HIGHLIGHT);
+        TRACE("SysLink Ctrl creation, hwnd=%p\n", hwnd);
+        lParam = (LPARAM)(((LPCREATESTRUCTW)lParam)->lpszName);
+        SYSLINK_SetText(infoPtr, (LPWSTR)lParam);
+        return 0;
+
+    case WM_DESTROY:
+        TRACE("SysLink Ctrl destruction, hwnd=%p\n", hwnd);
+        SYSLINK_ClearDoc(infoPtr);
+        SYSLINK_Free (infoPtr);
+        SetWindowLongW(hwnd, 0, 0);
+        return 0;
+
+    default:
+        if ((message >= WM_USER) && (message < WM_APP))
+	    ERR("unknown msg %04x wp=%04x lp=%08lx\n", message, wParam, lParam );
+        return DefWindowProcW(hwnd, message, wParam, lParam);
+    }
+}
+
+
+/***********************************************************************
+ * SYSLINK_Register [Internal]
+ *
+ * Registers the SysLink window class.
+ */
+VOID SYSLINK_Register (void)
+{
+    WNDCLASSW wndClass;
+
+    ZeroMemory (&wndClass, sizeof(wndClass));
+    wndClass.style         = CS_GLOBALCLASS | CS_VREDRAW | CS_HREDRAW;
+    wndClass.lpfnWndProc   = (WNDPROC)SysLinkWindowProc;
+    wndClass.cbClsExtra    = 0;
+    wndClass.cbWndExtra    = sizeof (SYSLINK_INFO *);
+    wndClass.hCursor       = LoadCursorW (0, (LPWSTR)IDC_ARROW);
+    wndClass.lpszClassName = WC_LINK;
+
+    RegisterClassW (&wndClass);
+}
+
+
+/***********************************************************************
+ * SYSLINK_Unregister [Internal]
+ *
+ * Unregisters the SysLink window class.
+ */
+VOID SYSLINK_Unregister (void)
+{
+    UnregisterClassW (WC_LINK, NULL);
+}
diff --git a/include/commctrl.h b/include/commctrl.h
index 278f476..b8470dd 100644
--- a/include/commctrl.h
+++ b/include/commctrl.h
@@ -4505,6 +4505,67 @@
 
 BOOL WINAPI Str_SetPtrW (LPWSTR *, LPCWSTR);
 
+/**************************************************************************
+ * SysLink control
+ */
+
+#if defined(__GNUC__)
+# define WC_LINK (const WCHAR []){ 'S','y','s','L','i','n','k',0 }
+#elif defined(_MSC_VER)
+# define WC_LINK             L"SysLink"
+#else
+static const WCHAR WC_LINK[] = { 'S','y','s','L','i','n','k',0 };
+#endif
+
+/* SysLink messages */
+#define LM_HITTEST           (WM_USER + 768)
+#define LM_GETIDEALHEIGHT    (WM_USER + 769)
+#define LM_SETITEM           (WM_USER + 770)
+#define LM_GETITEM           (WM_USER + 771)
+
+/* SysLink links flags */
+
+#define LIF_ITEMINDEX   1
+#define LIF_STATE       2
+#define LIF_ITEMID      4
+#define LIF_URL         8
+
+/* SysLink links states */
+
+#define LIS_FOCUSED     1
+#define LIS_ENABLED     2
+#define LIS_VISITED     4
+
+/* SysLink misc. */
+
+#define INVALID_LINK_INDEX  (-1)
+#define MAX_LINKID_TEXT     48
+#define L_MAX_URL_LENGTH    2084
+
+/* SysLink structures */
+
+typedef struct tagLITEM
+{
+  UINT mask;
+  int iLink;
+  UINT state;
+  UINT stateMask;
+  WCHAR szID[MAX_LINKID_TEXT];
+  WCHAR szUrl[L_MAX_URL_LENGTH];
+} LITEM, *PLITEM;
+
+typedef struct tagLHITTESTINFO
+{
+  POINT pt;
+  LITEM item;
+} LHITTESTINFO, *PLHITTESTINFO;
+
+typedef struct tagNMLINK
+{
+  NMHDR hdr;
+  LITEM item ;
+} NMLINK, *PNMLINK;
+
 #ifdef __cplusplus
 }
 #endif