| // 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 ( |
| "bufio" |
| "bytes" |
| "fmt" |
| "html/template" |
| "path/filepath" |
| "regexp" |
| "strconv" |
| "strings" |
| ) |
| |
| // PlayEnabled specifies whether runnable playground snippets should be |
| // displayed in the present user interface. |
| var PlayEnabled = false |
| |
| // TODO(adg): replace the PlayEnabled flag with something less spaghetti-like. |
| // Instead this will probably be determined by a template execution Context |
| // value that contains various global metadata required when rendering |
| // templates. |
| |
| // NotesEnabled specifies whether presenter notes should be displayed in the |
| // present user interface. |
| var NotesEnabled = false |
| |
| func init() { |
| Register("code", parseCode) |
| Register("play", parseCode) |
| } |
| |
| type Code struct { |
| Cmd string // original command from present source |
| Text template.HTML |
| Play bool // runnable code |
| Edit bool // editable code |
| FileName string // file name |
| Ext string // file extension |
| Raw []byte // content of the file |
| } |
| |
| func (c Code) PresentCmd() string { return c.Cmd } |
| func (c Code) TemplateName() string { return "code" } |
| |
| // The input line is a .code or .play entry with a file name and an optional HLfoo marker on the end. |
| // Anything between the file and HL (if any) is an address expression, which we treat as a string here. |
| // We pick off the HL first, for easy parsing. |
| var ( |
| highlightRE = regexp.MustCompile(`\s+HL([a-zA-Z0-9_]+)?$`) |
| hlCommentRE = regexp.MustCompile(`(.+) // HL(.*)$`) |
| codeRE = regexp.MustCompile(`\.(code|play)\s+((?:(?:-edit|-numbers)\s+)*)([^\s]+)(?:\s+(.*))?$`) |
| ) |
| |
| // parseCode parses a code present directive. Its syntax: |
| // |
| // .code [-numbers] [-edit] <filename> [address] [highlight] |
| // |
| // The directive may also be ".play" if the snippet is executable. |
| func parseCode(ctx *Context, sourceFile string, sourceLine int, cmd string) (Elem, error) { |
| cmd = strings.TrimSpace(cmd) |
| origCmd := cmd |
| |
| // Pull off the HL, if any, from the end of the input line. |
| highlight := "" |
| if hl := highlightRE.FindStringSubmatchIndex(cmd); len(hl) == 4 { |
| if hl[2] < 0 || hl[3] < 0 { |
| return nil, fmt.Errorf("%s:%d invalid highlight syntax", sourceFile, sourceLine) |
| } |
| highlight = cmd[hl[2]:hl[3]] |
| cmd = cmd[:hl[2]-2] |
| } |
| |
| // Parse the remaining command line. |
| // Arguments: |
| // args[0]: whole match |
| // args[1]: .code/.play |
| // args[2]: flags ("-edit -numbers") |
| // args[3]: file name |
| // args[4]: optional address |
| args := codeRE.FindStringSubmatch(cmd) |
| if len(args) != 5 { |
| return nil, fmt.Errorf("%s:%d: syntax error for .code/.play invocation", sourceFile, sourceLine) |
| } |
| command, flags, file, addr := args[1], args[2], args[3], strings.TrimSpace(args[4]) |
| play := command == "play" && PlayEnabled |
| |
| // Read in code file and (optionally) match address. |
| filename := filepath.Join(filepath.Dir(sourceFile), file) |
| textBytes, err := ctx.ReadFile(filename) |
| if err != nil { |
| return nil, fmt.Errorf("%s:%d: %v", sourceFile, sourceLine, err) |
| } |
| lo, hi, err := addrToByteRange(addr, 0, textBytes) |
| if err != nil { |
| return nil, fmt.Errorf("%s:%d: %v", sourceFile, sourceLine, err) |
| } |
| if lo > hi { |
| // The search in addrToByteRange can wrap around so we might |
| // end up with the range ending before its starting point |
| hi, lo = lo, hi |
| } |
| |
| // Acme pattern matches can stop mid-line, |
| // so run to end of line in both directions if not at line start/end. |
| for lo > 0 && textBytes[lo-1] != '\n' { |
| lo-- |
| } |
| if hi > 0 { |
| for hi < len(textBytes) && textBytes[hi-1] != '\n' { |
| hi++ |
| } |
| } |
| |
| lines := codeLines(textBytes, lo, hi) |
| |
| data := &codeTemplateData{ |
| Lines: formatLines(lines, highlight), |
| Edit: strings.Contains(flags, "-edit"), |
| Numbers: strings.Contains(flags, "-numbers"), |
| } |
| |
| // Include before and after in a hidden span for playground code. |
| if play { |
| data.Prefix = textBytes[:lo] |
| data.Suffix = textBytes[hi:] |
| } |
| |
| var buf bytes.Buffer |
| if err := codeTemplate.Execute(&buf, data); err != nil { |
| return nil, err |
| } |
| return Code{ |
| Cmd: origCmd, |
| Text: template.HTML(buf.String()), |
| Play: play, |
| Edit: data.Edit, |
| FileName: filepath.Base(filename), |
| Ext: filepath.Ext(filename), |
| Raw: rawCode(lines), |
| }, nil |
| } |
| |
| // formatLines returns a new slice of codeLine with the given lines |
| // replacing tabs with spaces and adding highlighting where needed. |
| func formatLines(lines []codeLine, highlight string) []codeLine { |
| formatted := make([]codeLine, len(lines)) |
| for i, line := range lines { |
| // Replace tabs with spaces, which work better in HTML. |
| line.L = strings.Replace(line.L, "\t", " ", -1) |
| |
| // Highlight lines that end with "// HL[highlight]" |
| // and strip the magic comment. |
| if m := hlCommentRE.FindStringSubmatch(line.L); m != nil { |
| line.L = m[1] |
| line.HL = m[2] == highlight |
| } |
| |
| formatted[i] = line |
| } |
| return formatted |
| } |
| |
| // rawCode returns the code represented by the given codeLines without any kind |
| // of formatting. |
| func rawCode(lines []codeLine) []byte { |
| b := new(bytes.Buffer) |
| for _, line := range lines { |
| b.WriteString(line.L) |
| b.WriteByte('\n') |
| } |
| return b.Bytes() |
| } |
| |
| type codeTemplateData struct { |
| Lines []codeLine |
| Prefix, Suffix []byte |
| Edit, Numbers bool |
| } |
| |
| var leadingSpaceRE = regexp.MustCompile(`^[ \t]*`) |
| |
| var codeTemplate = template.Must(template.New("code").Funcs(template.FuncMap{ |
| "trimSpace": strings.TrimSpace, |
| "leadingSpace": leadingSpaceRE.FindString, |
| }).Parse(codeTemplateHTML)) |
| |
| const codeTemplateHTML = ` |
| {{with .Prefix}}<pre style="display: none"><span>{{printf "%s" .}}</span></pre>{{end -}} |
| |
| <pre{{if .Edit}} contenteditable="true" spellcheck="false"{{end}}{{if .Numbers}} class="numbers"{{end}}>{{/* |
| */}}{{range .Lines}}<span num="{{.N}}">{{/* |
| */}}{{if .HL}}{{leadingSpace .L}}<b>{{trimSpace .L}}</b>{{/* |
| */}}{{else}}{{.L}}{{end}}{{/* |
| */}}</span> |
| {{end}}</pre> |
| {{with .Suffix}}<pre style="display: none"><span>{{printf "%s" .}}</span></pre>{{end -}} |
| ` |
| |
| // codeLine represents a line of code extracted from a source file. |
| type codeLine struct { |
| L string // The line of code. |
| N int // The line number from the source file. |
| HL bool // Whether the line should be highlighted. |
| } |
| |
| // codeLines takes a source file and returns the lines that |
| // span the byte range specified by start and end. |
| // It discards lines that end in "OMIT". |
| func codeLines(src []byte, start, end int) (lines []codeLine) { |
| startLine := 1 |
| for i, b := range src { |
| if i == start { |
| break |
| } |
| if b == '\n' { |
| startLine++ |
| } |
| } |
| s := bufio.NewScanner(bytes.NewReader(src[start:end])) |
| for n := startLine; s.Scan(); n++ { |
| l := s.Text() |
| if strings.HasSuffix(l, "OMIT") { |
| continue |
| } |
| lines = append(lines, codeLine{L: l, N: n}) |
| } |
| // Trim leading and trailing blank lines. |
| for len(lines) > 0 && len(lines[0].L) == 0 { |
| lines = lines[1:] |
| } |
| for len(lines) > 0 && len(lines[len(lines)-1].L) == 0 { |
| lines = lines[:len(lines)-1] |
| } |
| return |
| } |
| |
| func parseArgs(name string, line int, args []string) (res []interface{}, err error) { |
| res = make([]interface{}, len(args)) |
| for i, v := range args { |
| if len(v) == 0 { |
| return nil, fmt.Errorf("%s:%d bad code argument %q", name, line, v) |
| } |
| switch v[0] { |
| case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9': |
| n, err := strconv.Atoi(v) |
| if err != nil { |
| return nil, fmt.Errorf("%s:%d bad code argument %q", name, line, v) |
| } |
| res[i] = n |
| case '/': |
| if len(v) < 2 || v[len(v)-1] != '/' { |
| return nil, fmt.Errorf("%s:%d bad code argument %q", name, line, v) |
| } |
| res[i] = v |
| case '$': |
| res[i] = "$" |
| case '_': |
| if len(v) == 1 { |
| // Do nothing; "_" indicates an intentionally empty parameter. |
| break |
| } |
| fallthrough |
| default: |
| return nil, fmt.Errorf("%s:%d bad code argument %q", name, line, v) |
| } |
| } |
| return |
| } |