(
((http|https|ftp)://([\w-\d]+\.)+[\w-\d]+){0,1} // Capture domain names or IP addresses
(/[\w~,;\-\./?%&+#=]*) // Capture paths, including relative
)
Rationale for this answer:
- The whole thing is grouped so you can pick out the entire URL
- The protocol portion is optional, but if provided, a hostname or IP address should also be provided (both of which have fewer allowed characters than the rest of the URI).
- The "/" at the beginning is also optional. Paths can be in the form "images/1.gif", which are relative to the current path rather than relative to the hostname.
Caveats:
- mailto and file URIs not supported.
- URLs trailed by a period (such as at the end of a sentence without quotation) will include the trailing period.
- Because of #3 above, it's going to capture all sorts of things. If you can verify that all paths are not relative, you can add a "/" outside the parenthesis and thus require it.
- If all URIs are within HTML attributes (A, LINK, IMG, etc.), you can target the URIs much more accurately by only capturing within quotes, or at least only within HTML tags.
Edit: whoops, fixed closing paren problem.