| /* | 
 |  * SysLink control | 
 |  * | 
 |  * Copyright 2004 - 2006 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
 |  * | 
 |  * NOTES | 
 |  * | 
 |  * This code was audited for completeness against the documented features | 
 |  * of Comctl32.dll version 6.0 on Apr. 4, 2005, by Dimitrie O. Paun. | 
 |  *  | 
 |  * Unless otherwise noted, we believe this code to be complete, as per | 
 |  * the specification mentioned above. | 
 |  * If you discover missing features, or bugs, please note them below. | 
 |  */ | 
 |  | 
 | #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/unicode.h" | 
 | #include "wine/debug.h" | 
 |  | 
 | WINE_DEFAULT_DEBUG_CHANNEL(syslink); | 
 |  | 
 | INT WINAPI StrCmpNIW(LPCWSTR,LPCWSTR,INT); | 
 |  | 
 | typedef struct | 
 | { | 
 |     int nChars; | 
 |     int nSkip; | 
 |     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 */ | 
 |     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 */ | 
 |         } Link; | 
 |         struct | 
 |         { | 
 |             UINT Dummy; | 
 |         } Text; | 
 |     } u; | 
 |     WCHAR Text[1];          /* Text of the document item */ | 
 | } DOC_ITEM, *PDOC_ITEM; | 
 |  | 
 | typedef struct | 
 | { | 
 |     HWND      Self;         /* The window handle for this control */ | 
 |     HWND      Notify;       /* The parent handle to receive notifications */ | 
 |     DWORD     Style;        /* Styles 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 */ | 
 |     WCHAR     BreakChar;    /* Break Character for the current font */ | 
 |     BOOL      IgnoreReturn; /* (infoPtr->Style & LWS_IGNORERETURN) on creation */ | 
 | } 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) | 
 |     { | 
 |         Free(DocItem->u.Link.szID); | 
 |         Free(DocItem->u.Link.szUrl); | 
 |     } | 
 |  | 
 |     /* we don't free Text because it's just a pointer to a character in the | 
 |        entire window text string */ | 
 |  | 
 |     Free(DocItem); | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  * SYSLINK_AppendDocItem | 
 |  * Create and append a new document item. | 
 |  */ | 
 | static PDOC_ITEM SYSLINK_AppendDocItem (SYSLINK_INFO *infoPtr, LPCWSTR Text, UINT textlen, | 
 |                                         SL_ITEM_TYPE type, PDOC_ITEM LastItem) | 
 | { | 
 |     PDOC_ITEM Item; | 
 |  | 
 |     textlen = min(textlen, strlenW(Text)); | 
 |     Item = Alloc(FIELD_OFFSET(DOC_ITEM, Text[textlen + 1])); | 
 |     if(Item == NULL) | 
 |     { | 
 |         ERR("Failed to alloc DOC_ITEM structure!\n"); | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     Item->Next = NULL; | 
 |     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); | 
 |      | 
 |     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, LPCWSTR Text) | 
 | { | 
 |     LPCWSTR current, textstart = NULL, linktext = NULL, firsttag = NULL; | 
 |     int taglen = 0, textlen = 0, linklen = 0, docitems = 0; | 
 |     PDOC_ITEM Last = NULL; | 
 |     SL_ITEM_TYPE CurrentType = slText; | 
 |     LPCWSTR lpID, lpUrl; | 
 |     UINT lenId, lenUrl; | 
 |  | 
 |     TRACE("(%p %s)\n", infoPtr, debugstr_w(Text)); | 
 |  | 
 |     for(current = Text; *current != 0;) | 
 |     { | 
 |         if(*current == '<') | 
 |         { | 
 |             if(!StrCmpNIW(current, SL_LINKOPEN, 2) && (CurrentType == slText)) | 
 |             { | 
 |                 BOOL ValidParam = FALSE, ValidLink = FALSE; | 
 |  | 
 |                 if(*(current + 2) == '>') | 
 |                 { | 
 |                     /* we just have to deal with a <a> tag */ | 
 |                     taglen = 3; | 
 |                     ValidLink = TRUE; | 
 |                     ValidParam = TRUE; | 
 |                     firsttag = current; | 
 |                     linklen = 0; | 
 |                     lpID = NULL; | 
 |                     lpUrl = NULL; | 
 |                 } | 
 |                 else if(*(current + 2) == infoPtr->BreakChar) | 
 |                 { | 
 |                     /* we expect parameters, parse them */ | 
 |                     LPCWSTR *CurrentParameter = NULL, tmp; | 
 |                     UINT *CurrentParameterLen = NULL; | 
 |  | 
 |                     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 | 
 |                          */ | 
 |                         if(*tmp == infoPtr->BreakChar) | 
 |                         { | 
 |                             /* we expect another parameter, do the whole thing again */ | 
 |                             taglen++; | 
 |                             tmp++; | 
 |                             goto CheckParameter; | 
 |                         } | 
 |                         else if(*tmp == '>') | 
 |                         { | 
 |                             /* the tag is being closed, we're done */ | 
 |                             ValidLink = TRUE; | 
 |                             taglen++; | 
 |                         } | 
 |                     } | 
 |                 } | 
 |                  | 
 |                 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(!(infoPtr->Style & WS_DISABLED)) | 
 |                         { | 
 |                             Last->u.Link.state |= LIS_ENABLED; | 
 |                         } | 
 |                         /* Copy the tag parameters */ | 
 |                         if(lpID != NULL) | 
 |                         { | 
 |                             nc = min(lenId, strlenW(lpID)); | 
 |                             nc = min(nc, MAX_LINKID_TEXT - 1); | 
 |                             Last->u.Link.szID = Alloc((nc + 1) * sizeof(WCHAR)); | 
 |                             if(Last->u.Link.szID != NULL) | 
 |                             { | 
 |                                 lstrcpynW(Last->u.Link.szID, lpID, nc + 1); | 
 |                             } | 
 |                         } | 
 |                         else | 
 |                             Last->u.Link.szID = NULL; | 
 |                         if(lpUrl != NULL) | 
 |                         { | 
 |                             nc = min(lenUrl, strlenW(lpUrl)); | 
 |                             nc = min(nc, L_MAX_URL_LENGTH - 1); | 
 |                             Last->u.Link.szUrl = Alloc((nc + 1) * sizeof(WCHAR)); | 
 |                             if(Last->u.Link.szUrl != NULL) | 
 |                             { | 
 |                                 lstrcpynW(Last->u.Link.szUrl, lpUrl, nc + 1); | 
 |                             } | 
 |                         } | 
 |                         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(!(infoPtr->Style & WS_DISABLED)) | 
 |             { | 
 |                 Last->u.Link.state |= LIS_ENABLED; | 
 |             } | 
 |             /* Copy the tag parameters */ | 
 |             if(lpID != NULL) | 
 |             { | 
 |                 nc = min(lenId, strlenW(lpID)); | 
 |                 nc = min(nc, MAX_LINKID_TEXT - 1); | 
 |                 Last->u.Link.szID = Alloc((nc + 1) * sizeof(WCHAR)); | 
 |                 if(Last->u.Link.szID != NULL) | 
 |                 { | 
 |                     lstrcpynW(Last->u.Link.szID, lpID, nc + 1); | 
 |                 } | 
 |             } | 
 |             else | 
 |                 Last->u.Link.szID = NULL; | 
 |             if(lpUrl != NULL) | 
 |             { | 
 |                 nc = min(lenUrl, strlenW(lpUrl)); | 
 |                 nc = min(nc, L_MAX_URL_LENGTH - 1); | 
 |                 Last->u.Link.szUrl = Alloc((nc + 1) * sizeof(WCHAR)); | 
 |                 if(Last->u.Link.szUrl != NULL) | 
 |                 { | 
 |                     lstrcpynW(Last->u.Link.szUrl, lpUrl, nc + 1); | 
 |                 } | 
 |             } | 
 |             else | 
 |                 Last->u.Link.szUrl = NULL; | 
 |         } | 
 |         docitems++; | 
 |     } | 
 |  | 
 |     if(linktext != NULL && linklen > 0) | 
 |     { | 
 |         /* we got an 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 (const SYSLINK_INFO *infoPtr, const DOC_ITEM *DocItem) | 
 | { | 
 |     PDOC_TEXTBLOCK bl; | 
 |     int n; | 
 |  | 
 |     if(DocItem->Type != slLink) | 
 |     { | 
 |         ERR("DocItem not a link!\n"); | 
 |         return; | 
 |     } | 
 |      | 
 |     bl = DocItem->Blocks; | 
 |     if (bl != NULL) | 
 |     { | 
 |         n = DocItem->nText; | 
 |          | 
 |         while(n > 0) | 
 |         { | 
 |             InvalidateRect(infoPtr->Self, &bl->rc, TRUE); | 
 |             n -= bl->nChars + bl->nSkip; | 
 |             bl++; | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  * SYSLINK_GetLinkItemByIndex | 
 |  * Retrieves a document link by its index | 
 |  */ | 
 | static PDOC_ITEM SYSLINK_GetLinkItemByIndex (const 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 | 
 |  * Retrieves the link that has the LIS_FOCUSED bit | 
 |  */ | 
 | static PDOC_ITEM SYSLINK_GetFocusLink (const 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 (const 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 (const 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 (LPWSTR Text, WCHAR BreakChar, int x, int *LineLen, | 
 |                              int nFit, LPSIZE Extent) | 
 | { | 
 |     int i; | 
 |  | 
 |     for (i = 0; i < nFit; i++) if (Text[i] == '\n') break; | 
 |  | 
 |     if (i == *LineLen) return FALSE; | 
 |  | 
 |     /* check if we're in the middle of a word */ | 
 |     if (Text[i] != '\n' && Text[i] != BreakChar) | 
 |     { | 
 |         /* search for the beginning of the word */ | 
 |         while (i && Text[i - 1] != BreakChar) i--; | 
 |  | 
 |         if (i == 0) | 
 |         { | 
 |             Extent->cx = 0; | 
 |             Extent->cy = 0; | 
 |             if (x == SL_LEFTMARGIN) i = max( nFit, 1 ); | 
 |         } | 
 |     } | 
 |     *LineLen = i; | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  * SYSLINK_Render | 
 |  * Renders the document in memory | 
 |  */ | 
 | static VOID SYSLINK_Render (const SYSLINK_INFO *infoPtr, HDC hdc, PRECT pRect) | 
 | { | 
 |     RECT rc; | 
 |     PDOC_ITEM Current; | 
 |     HGDIOBJ hOldFont; | 
 |     int x, y, LineHeight; | 
 |     SIZE szDoc; | 
 |     TEXTMETRICW tm; | 
 |  | 
 |     szDoc.cx = szDoc.cy = 0; | 
 |  | 
 |     rc = *pRect; | 
 |     rc.right -= SL_RIGHTMARGIN; | 
 |     rc.bottom -= SL_BOTTOMMARGIN; | 
 |  | 
 |     if(rc.right - SL_LEFTMARGIN < 0) | 
 |         rc.right = MAXLONG; | 
 |     if (rc.bottom - SL_TOPMARGIN < 0) | 
 |         rc.bottom = MAXLONG; | 
 |      | 
 |     hOldFont = SelectObject(hdc, infoPtr->Font); | 
 |      | 
 |     x = SL_LEFTMARGIN; | 
 |     y = SL_TOPMARGIN; | 
 |     GetTextMetricsW( hdc, &tm ); | 
 |     LineHeight = tm.tmHeight + tm.tmExternalLeading; | 
 |  | 
 |     for(Current = infoPtr->Items; Current != NULL; Current = Current->Next) | 
 |     { | 
 |         int n, nBlocks; | 
 |         LPWSTR tx; | 
 |         PDOC_TEXTBLOCK bl, cbl; | 
 |         INT nFit; | 
 |         SIZE szDim; | 
 |         int SkipChars = 0; | 
 |  | 
 |         if(Current->nText == 0) | 
 |         { | 
 |             continue; | 
 |         } | 
 |  | 
 |         tx = Current->Text; | 
 |         n = Current->nText; | 
 |  | 
 |         Free(Current->Blocks); | 
 |         Current->Blocks = NULL; | 
 |         bl = NULL; | 
 |         nBlocks = 0; | 
 |  | 
 |         if(Current->Type == slText) | 
 |         { | 
 |             SelectObject(hdc, infoPtr->Font); | 
 |         } | 
 |         else if(Current->Type == slLink) | 
 |         { | 
 |             SelectObject(hdc, infoPtr->LinkFont); | 
 |         } | 
 |          | 
 |         while(n > 0) | 
 |         { | 
 |             /* skip break characters unless they're the first of the doc item */ | 
 |             if(tx != Current->Text || x == SL_LEFTMARGIN) | 
 |             { | 
 |                 if (n && *tx == '\n') | 
 |                 { | 
 |                     tx++; | 
 |                     SkipChars++; | 
 |                     n--; | 
 |                 } | 
 |                 while(n > 0 && (*tx) == infoPtr->BreakChar) | 
 |                 { | 
 |                     tx++; | 
 |                     SkipChars++; | 
 |                     n--; | 
 |                 } | 
 |             } | 
 |  | 
 |             if((n == 0 && SkipChars != 0) || | 
 |                GetTextExtentExPointW(hdc, tx, n, rc.right - x, &nFit, NULL, &szDim)) | 
 |             { | 
 |                 int LineLen = n; | 
 |                 BOOL Wrap = FALSE; | 
 |                 PDOC_TEXTBLOCK nbl; | 
 |                  | 
 |                 if(n != 0) | 
 |                 { | 
 |                     Wrap = SYSLINK_WrapLine(tx, infoPtr->BreakChar, x, &LineLen, nFit, &szDim); | 
 |  | 
 |                     if(LineLen == 0) | 
 |                     { | 
 |                         /* move one line down, the word didn't fit into the line */ | 
 |                         x = SL_LEFTMARGIN; | 
 |                         y += LineHeight; | 
 |                         continue; | 
 |                     } | 
 |  | 
 |                     if(LineLen != n) | 
 |                     { | 
 |                         if(!GetTextExtentExPointW(hdc, tx, LineLen, rc.right - x, NULL, NULL, &szDim)) | 
 |                         { | 
 |                             if(bl != NULL) | 
 |                             { | 
 |                                 Free(bl); | 
 |                                 bl = NULL; | 
 |                                 nBlocks = 0; | 
 |                             } | 
 |                             break; | 
 |                         } | 
 |                     } | 
 |                 } | 
 |                  | 
 |                 nbl = ReAlloc(bl, (nBlocks + 1) * sizeof(DOC_TEXTBLOCK)); | 
 |                 if (nbl != NULL) | 
 |                 { | 
 |                     bl = nbl; | 
 |                     nBlocks++; | 
 |  | 
 |                     cbl = bl + nBlocks - 1; | 
 |                      | 
 |                     cbl->nChars = LineLen; | 
 |                     cbl->nSkip = SkipChars; | 
 |                     cbl->rc.left = x; | 
 |                     cbl->rc.top = y; | 
 |                     cbl->rc.right = x + szDim.cx; | 
 |                     cbl->rc.bottom = y + szDim.cy; | 
 |  | 
 |                     if (cbl->rc.right > szDoc.cx) | 
 |                         szDoc.cx = cbl->rc.right; | 
 |                     if (cbl->rc.bottom > szDoc.cy) | 
 |                         szDoc.cy = cbl->rc.bottom; | 
 |  | 
 |                     if(LineLen != 0) | 
 |                     { | 
 |                         x += szDim.cx; | 
 |                         if(Wrap) | 
 |                         { | 
 |                             x = SL_LEFTMARGIN; | 
 |                             y += LineHeight; | 
 |                         } | 
 |                     } | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     Free(bl); | 
 |                     bl = NULL; | 
 |                     nBlocks = 0; | 
 |  | 
 |                     ERR("Failed to alloc DOC_TEXTBLOCK structure!\n"); | 
 |                     break; | 
 |                 } | 
 |                 n -= LineLen; | 
 |                 tx += LineLen; | 
 |                 SkipChars = 0; | 
 |             } | 
 |             else | 
 |             { | 
 |                 n--; | 
 |             } | 
 |         } | 
 |  | 
 |         if(nBlocks != 0) | 
 |         { | 
 |             Current->Blocks = bl; | 
 |         } | 
 |     } | 
 |      | 
 |     SelectObject(hdc, hOldFont); | 
 |  | 
 |     pRect->right = pRect->left + szDoc.cx; | 
 |     pRect->bottom = pRect->top + szDoc.cy; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  * SYSLINK_Draw | 
 |  * Draws the SysLink control. | 
 |  */ | 
 | static LRESULT SYSLINK_Draw (const SYSLINK_INFO *infoPtr, HDC hdc) | 
 | { | 
 |     RECT rc; | 
 |     PDOC_ITEM Current; | 
 |     HFONT hOldFont; | 
 |     COLORREF OldTextColor, OldBkColor; | 
 |     HBRUSH hBrush; | 
 |     UINT text_flags = ETO_CLIPPED; | 
 |     UINT mode = GetBkMode( hdc ); | 
 |  | 
 |     hOldFont = SelectObject(hdc, infoPtr->Font); | 
 |     OldTextColor = SetTextColor(hdc, infoPtr->TextColor); | 
 |     OldBkColor = SetBkColor(hdc, comctl32_color.clrWindow); | 
 |  | 
 |     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; | 
 |  | 
 |     hBrush = (HBRUSH)SendMessageW(infoPtr->Notify, WM_CTLCOLORSTATIC, | 
 |                                   (WPARAM)hdc, (LPARAM)infoPtr->Self); | 
 |     if (!(infoPtr->Style & LWS_TRANSPARENT)) | 
 |     { | 
 |         FillRect(hdc, &rc, hBrush); | 
 |         if (GetBkMode( hdc ) == OPAQUE) text_flags |= ETO_OPAQUE; | 
 |     } | 
 |     else SetBkMode( hdc, TRANSPARENT ); | 
 |  | 
 |     DeleteObject(hBrush); | 
 |  | 
 |     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) | 
 |             { | 
 |                 tx += bl->nSkip; | 
 |                 ExtTextOutW(hdc, bl->rc.left, bl->rc.top, text_flags, &bl->rc, tx, bl->nChars, NULL); | 
 |                 if((Current->Type == slLink) && (Current->u.Link.state & LIS_FOCUSED) && infoPtr->HasFocus) | 
 |                 { | 
 |                     COLORREF PrevTextColor; | 
 |                     PrevTextColor = SetTextColor(hdc, infoPtr->TextColor); | 
 |                     DrawFocusRect(hdc, &bl->rc); | 
 |                     SetTextColor(hdc, PrevTextColor); | 
 |                 } | 
 |                 tx += bl->nChars; | 
 |                 n -= bl->nChars + bl->nSkip; | 
 |                 bl++; | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     SetBkColor(hdc, OldBkColor); | 
 |     SetTextColor(hdc, OldTextColor); | 
 |     SelectObject(hdc, hOldFont); | 
 |     SetBkMode(hdc, mode); | 
 |     return 0; | 
 | } | 
 |  | 
 |  | 
 | /*********************************************************************** | 
 |  * SYSLINK_Paint | 
 |  * Handles the WM_PAINT message. | 
 |  */ | 
 | static LRESULT SYSLINK_Paint (const SYSLINK_INFO *infoPtr, HDC hdcParam) | 
 | { | 
 |     HDC hdc; | 
 |     PAINTSTRUCT ps; | 
 |  | 
 |     hdc = hdcParam ? hdcParam : BeginPaint (infoPtr->Self, &ps); | 
 |     if (hdc) | 
 |     { | 
 |         SYSLINK_Draw (infoPtr, hdc); | 
 |         if (!hdcParam) 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; | 
 |     TEXTMETRICW tm; | 
 |     RECT rcClient; | 
 |     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 */ | 
 |     if (GetClientRect(infoPtr->Self, &rcClient)) | 
 |     { | 
 |         hdc = GetDC(infoPtr->Self); | 
 |         if(hdc != NULL) | 
 |         { | 
 |             /* create a new underline font */ | 
 |             if(GetTextMetricsW(hdc, &tm) && | 
 |                GetObjectW(infoPtr->Font, sizeof(LOGFONTW), &lf)) | 
 |             { | 
 |                 lf.lfUnderline = TRUE; | 
 |                 infoPtr->LinkFont = CreateFontIndirectW(&lf); | 
 |                 infoPtr->BreakChar = tm.tmBreakChar; | 
 |             } | 
 |             else | 
 |             { | 
 |                 ERR("Failed to create link font!\n"); | 
 |             } | 
 |  | 
 |             SYSLINK_Render(infoPtr, hdc, &rcClient); | 
 |             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, LPCWSTR Text) | 
 | { | 
 |     /* clear the document */ | 
 |     SYSLINK_ClearDoc(infoPtr); | 
 |  | 
 |     if(Text == NULL || *Text == 0) | 
 |     { | 
 |         return TRUE; | 
 |     } | 
 |  | 
 |     /* let's parse the string and create a document */ | 
 |     if(SYSLINK_ParseText(infoPtr, Text) > 0) | 
 |     { | 
 |         RECT rcClient; | 
 |  | 
 |         /* Render text position and word wrapping in memory */ | 
 |         if (GetClientRect(infoPtr->Self, &rcClient)) | 
 |         { | 
 |             HDC hdc = GetDC(infoPtr->Self); | 
 |             if (hdc != NULL) | 
 |             { | 
 |                 SYSLINK_Render(infoPtr, hdc, &rcClient); | 
 |                 ReleaseDC(infoPtr->Self, hdc); | 
 |  | 
 |                 InvalidateRect(infoPtr->Self, NULL, TRUE); | 
 |             } | 
 |         } | 
 |     } | 
 |      | 
 |     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 (const SYSLINK_INFO *infoPtr, const DOC_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 (const SYSLINK_INFO *infoPtr, const LITEM *Item) | 
 | { | 
 |     PDOC_ITEM di; | 
 |     int nc; | 
 |     PWSTR szId = NULL; | 
 |     PWSTR szUrl = NULL; | 
 |     BOOL Repaint = FALSE; | 
 |  | 
 |     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_ITEMID) | 
 |     { | 
 |         nc = min(lstrlenW(Item->szID), MAX_LINKID_TEXT - 1); | 
 |         szId = Alloc((nc + 1) * sizeof(WCHAR)); | 
 |         if(szId) | 
 |         { | 
 |             lstrcpynW(szId, Item->szID, nc + 1); | 
 |         } | 
 |         else | 
 |         { | 
 |             ERR("Unable to allocate memory for link id\n"); | 
 |             return FALSE; | 
 |         } | 
 |     } | 
 |  | 
 |     if(Item->mask & LIF_URL) | 
 |     { | 
 |         nc = min(lstrlenW(Item->szUrl), L_MAX_URL_LENGTH - 1); | 
 |         szUrl = Alloc((nc + 1) * sizeof(WCHAR)); | 
 |         if(szUrl) | 
 |         { | 
 |             lstrcpynW(szUrl, Item->szUrl, nc + 1); | 
 |         } | 
 |         else | 
 |         { | 
 |             Free(szId); | 
 |  | 
 |             ERR("Unable to allocate memory for link url\n"); | 
 |             return FALSE; | 
 |         } | 
 |     } | 
 |  | 
 |     if(Item->mask & LIF_ITEMID) | 
 |     { | 
 |         Free(di->u.Link.szID); | 
 |         di->u.Link.szID = szId; | 
 |     } | 
 |  | 
 |     if(Item->mask & LIF_URL) | 
 |     { | 
 |         Free(di->u.Link.szUrl); | 
 |         di->u.Link.szUrl = szUrl; | 
 |     } | 
 |  | 
 |     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(Repaint) | 
 |     { | 
 |         SYSLINK_RepaintLink(infoPtr, di); | 
 |     } | 
 |      | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_GetItem | 
 |  * Retrieves the states and attributes of a link item. | 
 |  */ | 
 | static LRESULT SYSLINK_GetItem (const 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) | 
 |         { | 
 |             lstrcpyW(Item->szID, di->u.Link.szID); | 
 |         } | 
 |         else | 
 |         { | 
 |             Item->szID[0] = 0; | 
 |         } | 
 |     } | 
 |      | 
 |     if(Item->mask & LIF_URL) | 
 |     { | 
 |         if(di->u.Link.szUrl) | 
 |         { | 
 |             lstrcpyW(Item->szUrl, di->u.Link.szUrl); | 
 |         } | 
 |         else | 
 |         { | 
 |             Item->szUrl[0] = 0; | 
 |         } | 
 |     } | 
 |      | 
 |     return TRUE; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_PtInDocItem | 
 |  * Determines if a point is in the region of a document item | 
 |  */ | 
 | static BOOL SYSLINK_PtInDocItem (const DOC_ITEM *DocItem, POINT pt) | 
 | { | 
 |     PDOC_TEXTBLOCK bl; | 
 |     int n; | 
 |  | 
 |     bl = DocItem->Blocks; | 
 |     if (bl != NULL) | 
 |     { | 
 |         n = DocItem->nText; | 
 |  | 
 |         while(n > 0) | 
 |         { | 
 |             if (PtInRect(&bl->rc, pt)) | 
 |             { | 
 |                 return TRUE; | 
 |             } | 
 |             n -= bl->nChars + bl->nSkip; | 
 |             bl++; | 
 |         } | 
 |     } | 
 |      | 
 |     return FALSE; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_HitTest | 
 |  * Determines the link the user clicked on. | 
 |  */ | 
 | static LRESULT SYSLINK_HitTest (const 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(SYSLINK_PtInDocItem(Current, HitTest->pt)) | 
 |             { | 
 |                 HitTest->item.mask = 0; | 
 |                 HitTest->item.iLink = id; | 
 |                 HitTest->item.state = 0; | 
 |                 HitTest->item.stateMask = 0; | 
 |                 if(Current->u.Link.szID) | 
 |                 { | 
 |                     lstrcpyW(HitTest->item.szID, Current->u.Link.szID); | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     HitTest->item.szID[0] = 0; | 
 |                 } | 
 |                 if(Current->u.Link.szUrl) | 
 |                 { | 
 |                     lstrcpyW(HitTest->item.szUrl, Current->u.Link.szUrl); | 
 |                 } | 
 |                 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 (const 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 (const SYSLINK_INFO *infoPtr, UINT code, const DOC_ITEM *Link, int iLink) | 
 | { | 
 |     NMLINK nml; | 
 |  | 
 |     nml.hdr.hwndFrom = infoPtr->Self; | 
 |     nml.hdr.idFrom = GetWindowLongPtrW(infoPtr->Self, GWLP_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) | 
 |     { | 
 |         lstrcpyW(nml.item.szID, Link->u.Link.szID); | 
 |     } | 
 |     else | 
 |     { | 
 |         nml.item.szID[0] = 0; | 
 |     } | 
 |     if(Link->u.Link.szUrl) | 
 |     { | 
 |         lstrcpyW(nml.item.szUrl, Link->u.Link.szUrl); | 
 |     } | 
 |     else | 
 |     { | 
 |         nml.item.szUrl[0] = 0; | 
 |     } | 
 |  | 
 |     return SendMessageW(infoPtr->Notify, WM_NOTIFY, nml.hdr.idFrom, (LPARAM)&nml); | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_SetFocus | 
 |  * Handles receiving the input focus. | 
 |  */ | 
 | static LRESULT SYSLINK_SetFocus (SYSLINK_INFO *infoPtr) | 
 | { | 
 |     PDOC_ITEM Focus; | 
 |      | 
 |     infoPtr->HasFocus = TRUE; | 
 |  | 
 |     /* We always select the first link, even if we activated the control using | 
 |        SHIFT+TAB. This is the default behavior */ | 
 |     Focus = SYSLINK_GetNextLink(infoPtr, NULL); | 
 |     if(Focus != NULL) | 
 |     { | 
 |         SYSLINK_SetFocusLink(infoPtr, Focus); | 
 |         SYSLINK_RepaintLink(infoPtr, Focus); | 
 |     } | 
 |     return 0; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_KillFocus | 
 |  * Handles losing the input focus. | 
 |  */ | 
 | static LRESULT SYSLINK_KillFocus (SYSLINK_INFO *infoPtr) | 
 | { | 
 |     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 (const SYSLINK_INFO *infoPtr, const 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) && SYSLINK_PtInDocItem(Current, *pt) && | 
 |            (!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, const POINT *pt) | 
 | { | 
 |     PDOC_ITEM Current, Old; | 
 |     int id; | 
 |  | 
 |     Current = SYSLINK_LinkAtPt(infoPtr, pt, &id, TRUE); | 
 |     if(Current != NULL) | 
 |     { | 
 |       SetFocus(infoPtr->Self); | 
 |  | 
 |       Old = SYSLINK_SetFocusLink(infoPtr, Current); | 
 |       if(Old != NULL && Old != Current) | 
 |       { | 
 |           SYSLINK_RepaintLink(infoPtr, Old); | 
 |       } | 
 |       infoPtr->MouseDownID = id; | 
 |       SYSLINK_RepaintLink(infoPtr, Current); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_LButtonUp | 
 |  * Handles mouse clicks | 
 |  */ | 
 | static LRESULT SYSLINK_LButtonUp (SYSLINK_INFO *infoPtr, const 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 (const SYSLINK_INFO *infoPtr) | 
 | { | 
 |     if(infoPtr->HasFocus && !infoPtr->IgnoreReturn) | 
 |     { | 
 |         PDOC_ITEM Focus; | 
 |         int id; | 
 |          | 
 |         Focus = SYSLINK_GetFocusLink(infoPtr, &id); | 
 |         if(Focus) | 
 |         { | 
 |             SYSLINK_SendParentNotify(infoPtr, NM_RETURN, Focus, id); | 
 |             return TRUE; | 
 |         } | 
 |     } | 
 |     return FALSE; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSKEY_SelectNextPrevLink | 
 |  * Changes the currently focused link | 
 |  */ | 
 | static BOOL SYSKEY_SelectNextPrevLink (const 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 && 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 (const 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; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SYSLINK_GetIdealSize | 
 |  * Calculates the ideal size of a link control at a given maximum width. | 
 |  */ | 
 | static VOID SYSLINK_GetIdealSize (const SYSLINK_INFO *infoPtr, int cxMaxWidth, LPSIZE lpSize) | 
 | { | 
 |     RECT rc; | 
 |     HDC hdc; | 
 |  | 
 |     rc.left = rc.top = rc.bottom = 0; | 
 |     rc.right = cxMaxWidth; | 
 |  | 
 |     hdc = GetDC(infoPtr->Self); | 
 |     if (hdc != NULL) | 
 |     { | 
 |         HGDIOBJ hOldFont = SelectObject(hdc, infoPtr->Font); | 
 |  | 
 |         SYSLINK_Render(infoPtr, hdc, &rc); | 
 |  | 
 |         SelectObject(hdc, hOldFont); | 
 |         ReleaseDC(infoPtr->Self, hdc); | 
 |  | 
 |         lpSize->cx = rc.right; | 
 |         lpSize->cy = rc.bottom; | 
 |     } | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           SysLinkWindowProc | 
 |  */ | 
 | static LRESULT WINAPI SysLinkWindowProc(HWND hwnd, UINT message, | 
 |                                         WPARAM wParam, LPARAM lParam) | 
 | { | 
 |     SYSLINK_INFO *infoPtr; | 
 |  | 
 |     TRACE("hwnd=%p msg=%04x wparam=%lx lParam=%lx\n", hwnd, message, wParam, lParam); | 
 |  | 
 |     infoPtr = (SYSLINK_INFO *)GetWindowLongPtrW(hwnd, 0); | 
 |  | 
 |     if (!infoPtr && message != WM_CREATE) | 
 |         return DefWindowProcW(hwnd, message, wParam, lParam); | 
 |  | 
 |     switch(message) { | 
 |     case WM_PRINTCLIENT: | 
 |     case WM_PAINT: | 
 |         return SYSLINK_Paint (infoPtr, (HDC)wParam); | 
 |  | 
 |     case WM_ERASEBKGND: | 
 |         if (!(infoPtr->Style & LWS_TRANSPARENT)) | 
 |         { | 
 |             HDC hdc = (HDC)wParam; | 
 |             HBRUSH brush = CreateSolidBrush( comctl32_color.clrWindow ); | 
 |             RECT rect; | 
 |  | 
 |             GetClipBox( hdc, &rect ); | 
 |             FillRect( hdc, &rect, brush ); | 
 |             DeleteObject( brush ); | 
 |             return 1; | 
 |         } | 
 |         return 0; | 
 |  | 
 |     case WM_SETCURSOR: | 
 |     { | 
 |         LHITTESTINFO ht; | 
 |         DWORD mp = GetMessagePos(); | 
 |          | 
 |         ht.pt.x = (short)LOWORD(mp); | 
 |         ht.pt.y = (short)HIWORD(mp); | 
 |          | 
 |         ScreenToClient(infoPtr->Self, &ht.pt); | 
 |         if(SYSLINK_HitTest (infoPtr, &ht)) | 
 |         { | 
 |             SetCursor(LoadCursorW(0, (LPCWSTR)IDC_HAND)); | 
 |             return TRUE; | 
 |         } | 
 |  | 
 |         return DefWindowProcW(hwnd, message, wParam, lParam); | 
 |     } | 
 |  | 
 |     case WM_SIZE: | 
 |     { | 
 |         RECT rcClient; | 
 |         if (GetClientRect(infoPtr->Self, &rcClient)) | 
 |         { | 
 |             HDC hdc = GetDC(infoPtr->Self); | 
 |             if(hdc != NULL) | 
 |             { | 
 |                 SYSLINK_Render(infoPtr, hdc, &rcClient); | 
 |                 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 = (short)LOWORD(lParam); | 
 |         pt.y = (short)HIWORD(lParam); | 
 |         return SYSLINK_LButtonDown(infoPtr, &pt); | 
 |     } | 
 |     case WM_LBUTTONUP: | 
 |     { | 
 |         POINT pt; | 
 |         pt.x = (short)LOWORD(lParam); | 
 |         pt.y = (short)HIWORD(lParam); | 
 |         return SYSLINK_LButtonUp(infoPtr, &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; | 
 |         } | 
 |         default: | 
 |             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 = (short)LOWORD(lParam); | 
 |         pt.y = (short)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: | 
 |         if (lParam) | 
 |         { | 
 |             /* LM_GETIDEALSIZE */ | 
 |             SYSLINK_GetIdealSize(infoPtr, (int)wParam, (LPSIZE)lParam); | 
 |         } | 
 |         return SYSLINK_GetIdealHeight(infoPtr); | 
 |  | 
 |     case WM_SETFOCUS: | 
 |         return SYSLINK_SetFocus(infoPtr); | 
 |  | 
 |     case WM_KILLFOCUS: | 
 |         return SYSLINK_KillFocus(infoPtr); | 
 |  | 
 |     case WM_ENABLE: | 
 |         infoPtr->Style &= ~WS_DISABLED; | 
 |         infoPtr->Style |= (wParam ? 0 : WS_DISABLED); | 
 |         InvalidateRect (infoPtr->Self, NULL, FALSE); | 
 |         return 0; | 
 |  | 
 |     case WM_STYLECHANGED: | 
 |         if (wParam == GWL_STYLE) | 
 |         { | 
 |             infoPtr->Style = ((LPSTYLESTRUCT)lParam)->styleNew; | 
 |  | 
 |             InvalidateRect(infoPtr->Self, NULL, TRUE); | 
 |         } | 
 |         return 0; | 
 |  | 
 |     case WM_CREATE: | 
 |         /* allocate memory for info struct */ | 
 |         infoPtr = Alloc (sizeof(SYSLINK_INFO)); | 
 |         if (!infoPtr) return -1; | 
 |         SetWindowLongPtrW (hwnd, 0, (DWORD_PTR)infoPtr); | 
 |  | 
 |         /* initialize the info struct */ | 
 |         infoPtr->Self = hwnd; | 
 |         infoPtr->Notify = ((LPCREATESTRUCTW)lParam)->hwndParent; | 
 |         infoPtr->Style = ((LPCREATESTRUCTW)lParam)->style; | 
 |         infoPtr->Font = 0; | 
 |         infoPtr->LinkFont = 0; | 
 |         infoPtr->Items = NULL; | 
 |         infoPtr->HasFocus = FALSE; | 
 |         infoPtr->MouseDownID = -1; | 
 |         infoPtr->TextColor = comctl32_color.clrWindowText; | 
 |         infoPtr->LinkColor = comctl32_color.clrHighlight; | 
 |         infoPtr->VisitedColor = comctl32_color.clrHighlight; | 
 |         infoPtr->BreakChar = ' '; | 
 |         infoPtr->IgnoreReturn = infoPtr->Style & LWS_IGNORERETURN; | 
 |         TRACE("SysLink Ctrl creation, hwnd=%p\n", hwnd); | 
 |         SYSLINK_SetText(infoPtr, ((LPCREATESTRUCTW)lParam)->lpszName); | 
 |         return 0; | 
 |  | 
 |     case WM_DESTROY: | 
 |         TRACE("SysLink Ctrl destruction, hwnd=%p\n", hwnd); | 
 |         SYSLINK_ClearDoc(infoPtr); | 
 |         if(infoPtr->Font != 0) DeleteObject(infoPtr->Font); | 
 |         if(infoPtr->LinkFont != 0) DeleteObject(infoPtr->LinkFont); | 
 |         SetWindowLongPtrW(hwnd, 0, 0); | 
 |         Free (infoPtr); | 
 |         return 0; | 
 |  | 
 |     case WM_SYSCOLORCHANGE: | 
 |         COMCTL32_RefreshSysColors(); | 
 |         return 0; | 
 |  | 
 |     default: | 
 |         if ((message >= WM_USER) && (message < WM_APP) && !COMCTL32_IsReflectedMessage(message)) | 
 |         { | 
 |             ERR("unknown msg %04x wp=%04lx 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   = 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); | 
 | } |