From 911fd9a7bdade5953302159bfaa066114197e6a1 Mon Sep 17 00:00:00 2001 From: dail8859 Date: Mon, 14 Nov 2016 17:45:42 -0500 Subject: [PATCH] Simplify determination of URL at click location Closes #1248, Closes #1255, Closes #1595, Closes #1762, Closes #1768, Closes #2583 --- PowerEditor/src/NppNotification.cpp | 60 ++++++++--------------------- 1 file changed, 17 insertions(+), 43 deletions(-) diff --git a/PowerEditor/src/NppNotification.cpp b/PowerEditor/src/NppNotification.cpp index 2b8feb670..714f28aa6 100644 --- a/PowerEditor/src/NppNotification.cpp +++ b/PowerEditor/src/NppNotification.cpp @@ -887,54 +887,28 @@ BOOL Notepad_plus::notify(SCNotification *notification) if (not notifyView) return FALSE; - // Save the current wordChars before setting a custom list - const size_t wordBufferSize = notifyView->execute(SCI_GETWORDCHARS); - char *wordChars = new char[wordBufferSize + 1]; - notifyView->execute(SCI_GETWORDCHARS, 0, reinterpret_cast(wordChars)); - wordChars[wordBufferSize] = '\0'; + // Get the style and make sure it is a hotspot + auto style = notifyView->execute(SCI_GETSTYLEAT, notification->position); + if (not notifyView->execute(SCI_STYLEGETHOTSPOT, style)) + break; - // Set a custom list for detecting links - notifyView->execute(SCI_SETWORDCHARS, 0, reinterpret_cast("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_-+.,:?&@=/%#()")); + int startPos, endPos, docLen; + startPos = endPos = notification->position; + docLen = notifyView->getCurrentDocLen(); - auto pos = notifyView->execute(SCI_GETCURRENTPOS); - int startPos = static_cast(notifyView->execute(SCI_WORDSTARTPOSITION, pos, false)); - int endPos = static_cast(notifyView->execute(SCI_WORDENDPOSITION, pos, false)); + // Walk backwards/forwards to get the contiguous text in the same style + while (startPos > 0 && notifyView->execute(SCI_GETSTYLEAT, startPos - 1) == style) + startPos--; + while (endPos < docLen && notifyView->execute(SCI_GETSTYLEAT, endPos) == style) + endPos++; - notifyView->execute(SCI_SETTARGETRANGE, startPos, endPos); + // Select the entire link + notifyView->execute(SCI_SETANCHOR, startPos); + notifyView->execute(SCI_SETCURRENTPOS, endPos); - int posFound = static_cast(notifyView->execute(SCI_SEARCHINTARGET, strlen(URL_REG_EXPR), reinterpret_cast(URL_REG_EXPR))); - if (posFound != -2) - { - if (posFound != -1) - { - startPos = int(notifyView->execute(SCI_GETTARGETSTART)); - endPos = int(notifyView->execute(SCI_GETTARGETEND)); - } + generic_string url = notifyView->getGenericTextAsString(startPos, endPos); + ::ShellExecute(_pPublicInterface->getHSelf(), TEXT("open"), url.c_str(), NULL, NULL, SW_SHOW); - // Prevent buffer overflow in getGenericText(). - if(endPos - startPos > 2*MAX_PATH) - endPos = startPos + 2*MAX_PATH; - - TCHAR currentWord[2*MAX_PATH]; - - notifyView->getGenericText(currentWord, MAX_PATH*2, startPos, endPos); - - // This treatment would fail on some valid URLs where there's actually supposed to be a comma or parenthesis at the end. - size_t lastCharIndex = _tcsnlen(currentWord, MAX_PATH*2) - 1; - if ((currentWord[lastCharIndex] == ',' || currentWord[lastCharIndex] == ')' || currentWord[lastCharIndex] == '(')) - currentWord[lastCharIndex] = '\0'; - - ::ShellExecute(_pPublicInterface->getHSelf(), TEXT("open"), currentWord, NULL, NULL, SW_SHOW); - - // Re-set the previous wordChar list - notifyView->execute(SCI_SETWORDCHARS, 0, reinterpret_cast(wordChars)); - - // Select the entire link - notifyView->execute(SCI_SETANCHOR, startPos); - notifyView->execute(SCI_SETCURRENTPOS, endPos); - } - - delete[] wordChars; break; }