|  | // Copyright 2012 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package present | 
|  |  | 
|  | import ( | 
|  | "fmt" | 
|  | "log" | 
|  | "net/url" | 
|  | "strings" | 
|  | ) | 
|  |  | 
|  | func init() { | 
|  | Register("link", parseLink) | 
|  | } | 
|  |  | 
|  | type Link struct { | 
|  | Cmd   string // original command from present source | 
|  | URL   *url.URL | 
|  | Label string | 
|  | } | 
|  |  | 
|  | func (l Link) PresentCmd() string   { return l.Cmd } | 
|  | func (l Link) TemplateName() string { return "link" } | 
|  |  | 
|  | func parseLink(ctx *Context, fileName string, lineno int, text string) (Elem, error) { | 
|  | args := strings.Fields(text) | 
|  | if len(args) < 2 { | 
|  | return nil, fmt.Errorf("link element must have at least 2 arguments") | 
|  | } | 
|  | url, err := url.Parse(args[1]) | 
|  | if err != nil { | 
|  | return nil, err | 
|  | } | 
|  | label := "" | 
|  | if len(args) > 2 { | 
|  | label = strings.Join(args[2:], " ") | 
|  | } else { | 
|  | scheme := url.Scheme + "://" | 
|  | if url.Scheme == "mailto" { | 
|  | scheme = "mailto:" | 
|  | } | 
|  | label = strings.Replace(url.String(), scheme, "", 1) | 
|  | } | 
|  | return Link{text, url, label}, nil | 
|  | } | 
|  |  | 
|  | func renderLink(href, text string) string { | 
|  | text = font(text) | 
|  | if text == "" { | 
|  | text = href | 
|  | } | 
|  | // Open links in new window only when their url is absolute. | 
|  | target := "_blank" | 
|  | if u, err := url.Parse(href); err != nil { | 
|  | log.Println("renderLink parsing url:", err) | 
|  | } else if !u.IsAbs() || u.Scheme == "javascript" { | 
|  | target = "_self" | 
|  | } | 
|  |  | 
|  | return fmt.Sprintf(`<a href="%s" target="%s">%s</a>`, href, target, text) | 
|  | } | 
|  |  | 
|  | // parseInlineLink parses an inline link at the start of s, and returns | 
|  | // a rendered HTML link and the total length of the raw inline link. | 
|  | // If no inline link is present, it returns all zeroes. | 
|  | func parseInlineLink(s string) (link string, length int) { | 
|  | if !strings.HasPrefix(s, "[[") { | 
|  | return | 
|  | } | 
|  | end := strings.Index(s, "]]") | 
|  | if end == -1 { | 
|  | return | 
|  | } | 
|  | urlEnd := strings.Index(s, "]") | 
|  | rawURL := s[2:urlEnd] | 
|  | const badURLChars = `<>"{}|\^[] ` + "`" // per RFC2396 section 2.4.3 | 
|  | if strings.ContainsAny(rawURL, badURLChars) { | 
|  | return | 
|  | } | 
|  | if urlEnd == end { | 
|  | simpleURL := "" | 
|  | url, err := url.Parse(rawURL) | 
|  | if err == nil { | 
|  | // If the URL is http://foo.com, drop the http:// | 
|  | // In other words, render [[http://golang.org]] as: | 
|  | //   <a href="http://golang.org">golang.org</a> | 
|  | if strings.HasPrefix(rawURL, url.Scheme+"://") { | 
|  | simpleURL = strings.TrimPrefix(rawURL, url.Scheme+"://") | 
|  | } else if strings.HasPrefix(rawURL, url.Scheme+":") { | 
|  | simpleURL = strings.TrimPrefix(rawURL, url.Scheme+":") | 
|  | } | 
|  | } | 
|  | return renderLink(rawURL, simpleURL), end + 2 | 
|  | } | 
|  | if s[urlEnd:urlEnd+2] != "][" { | 
|  | return | 
|  | } | 
|  | text := s[urlEnd+2 : end] | 
|  | return renderLink(rawURL, text), end + 2 | 
|  | } |