blob: bb4ae862615f8aa906082179844002a908b5d678 [file] [log] [blame]
Ted Kremenek6a340832008-03-18 21:19:49 +00001//== HTMLRewrite.cpp - Translate source code into prettified HTML --*- C++ -*-//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file defines the HTMLRewriter clas, which is used to translate the
11// text of a source file into prettified HTML.
12//
13//===----------------------------------------------------------------------===//
14
15#include "clang/Rewrite/Rewriter.h"
16#include "clang/Rewrite/HTMLRewrite.h"
17#include "clang/Basic/SourceManager.h"
18#include "llvm/Support/MemoryBuffer.h"
19#include <sstream>
20
21using namespace clang;
22
23void html::EscapeText(Rewriter& R, unsigned FileID, bool EscapeSpaces) {
24
25 const llvm::MemoryBuffer *Buf = R.getSourceMgr().getBuffer(FileID);
26 const char* C = Buf->getBufferStart();
27 const char* FileEnd = Buf->getBufferEnd();
28
29 assert (C <= FileEnd);
30
31 for (unsigned FilePos = 0; C != FileEnd ; ++C, ++FilePos) {
32
33 SourceLocation Loc = SourceLocation::getFileLoc(FileID, FilePos);
34
35 switch (*C) {
36 default: break;
37
38 case ' ':
39 if (EscapeSpaces) R.ReplaceText(Loc, 1, "&#32;", 5);
40 break;
41
42 case '<': R.ReplaceText(Loc, 1, "&lt;", 4); break;
43 case '>': R.ReplaceText(Loc, 1, "&gt;", 4); break;
44 case '&': R.ReplaceText(Loc, 1, "&amp;", 5); break;
45 }
46 }
47}
48
Ted Kremenek053ef592008-03-27 17:15:29 +000049std::string html::EscapeText(const std::string& s, bool EscapeSpaces) {
50
51 unsigned len = s.size();
52 std::ostringstream os;
53
54 for (unsigned i = 0 ; i < len; ++i) {
55
56 char c = s[i];
57
58 switch (c) {
59 default:
60 os << c; break;
61
62 case ' ':
63 if (EscapeSpaces) os << "&#32;";
64 break;
65
66 case '<': os << "&lt;"; break;
67 case '>': os << "&gt;"; break;
68 case '&': os << "&amp;"; break;
69 }
70 }
71
72 return os.str();
73}
74
Ted Kremenekb485cd12008-03-18 23:08:51 +000075static void AddLineNumber(Rewriter& R, unsigned LineNo,
76 SourceLocation B, SourceLocation E) {
Ted Kremenekf8309972008-03-19 01:30:02 +000077
Ted Kremeneke8e019e2008-03-19 21:59:05 +000078 // Put the closing </tr> first.
Ted Kremenek13e479b2008-03-19 07:53:42 +000079
Ted Kremeneke8e019e2008-03-19 21:59:05 +000080 R.InsertCStrBefore(E, "</tr>");
Ted Kremenekb485cd12008-03-18 23:08:51 +000081
Ted Kremenekd6c13602008-03-19 05:07:26 +000082 if (B == E) // Handle empty lines.
Ted Kremeneke8e019e2008-03-19 21:59:05 +000083 R.InsertCStrBefore(B, "<td class=\"line\"> </td>");
Ted Kremenekd6c13602008-03-19 05:07:26 +000084 else {
Ted Kremeneke8e019e2008-03-19 21:59:05 +000085 R.InsertCStrBefore(E, "</td>");
86 R.InsertCStrBefore(B, "<td class=\"line\">");
Ted Kremenekd6c13602008-03-19 05:07:26 +000087 }
Ted Kremenekf8309972008-03-19 01:30:02 +000088
Ted Kremenekd6c13602008-03-19 05:07:26 +000089 // Insert a div tag for the line number.
90
Ted Kremenekb485cd12008-03-18 23:08:51 +000091 std::ostringstream os;
Ted Kremeneke8e019e2008-03-19 21:59:05 +000092 os << "<td class=\"num\">" << LineNo << "</td>";
Ted Kremenekd6c13602008-03-19 05:07:26 +000093
94 R.InsertStrBefore(B, os.str());
95
Ted Kremeneke8e019e2008-03-19 21:59:05 +000096 // Now prepend the <tr>.
Ted Kremenekd6c13602008-03-19 05:07:26 +000097
Ted Kremeneke8e019e2008-03-19 21:59:05 +000098 R.InsertCStrBefore(B, "<tr>");
Ted Kremenek13e479b2008-03-19 07:53:42 +000099
Ted Kremenekb485cd12008-03-18 23:08:51 +0000100}
101
102void html::AddLineNumbers(Rewriter& R, unsigned FileID) {
103
104 const llvm::MemoryBuffer *Buf = R.getSourceMgr().getBuffer(FileID);
105 const char* FileBeg = Buf->getBufferStart();
106 const char* FileEnd = Buf->getBufferEnd();
107 const char* C = FileBeg;
108
109 assert (C <= FileEnd);
110
111 unsigned LineNo = 0;
112 unsigned FilePos = 0;
113
114 while (C != FileEnd) {
115
116 ++LineNo;
117 unsigned LineStartPos = FilePos;
118 unsigned LineEndPos = FileEnd - FileBeg;
119
120 assert (FilePos <= LineEndPos);
121 assert (C < FileEnd);
122
123 // Scan until the newline (or end-of-file).
124
125 for ( ; C != FileEnd ; ++C, ++FilePos)
126 if (*C == '\n') {
127 LineEndPos = FilePos;
128 break;
129 }
130
131 AddLineNumber(R, LineNo,
132 SourceLocation::getFileLoc(FileID, LineStartPos),
133 SourceLocation::getFileLoc(FileID, LineEndPos));
134
135 if (C != FileEnd) {
136 ++C;
137 ++FilePos;
138 }
Ted Kremenekd6c13602008-03-19 05:07:26 +0000139 }
140
141 // Add one big div tag that surrounds all of the code.
142
143 R.InsertCStrBefore(SourceLocation::getFileLoc(FileID, 0),
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000144 "<table class=\"code\">\n");
Ted Kremenekd6c13602008-03-19 05:07:26 +0000145
146 R.InsertCStrAfter(SourceLocation::getFileLoc(FileID, FileEnd - FileBeg),
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000147 "</table>");
Ted Kremenekb485cd12008-03-18 23:08:51 +0000148}
Ted Kremenekad0a2032008-03-19 06:14:37 +0000149
150void html::AddHeaderFooterInternalBuiltinCSS(Rewriter& R, unsigned FileID) {
151
152 const llvm::MemoryBuffer *Buf = R.getSourceMgr().getBuffer(FileID);
153 const char* FileStart = Buf->getBufferStart();
154 const char* FileEnd = Buf->getBufferEnd();
155
156 SourceLocation StartLoc = SourceLocation::getFileLoc(FileID, 0);
157 SourceLocation EndLoc = SourceLocation::getFileLoc(FileID, FileEnd-FileStart);
158
159 // Generate header
160
161 {
162 std::ostringstream os;
163
164 os << "<html>\n<head>\n"
Ted Kremenekeeacc522008-03-19 22:44:21 +0000165 << "<style type=\"text/css\">\n"
166 << " body { color:#000000; background-color:#ffffff }\n"
Ted Kremenek2e939812008-03-27 07:35:49 +0000167 << " body { font-family:Helvetica, sans-serif }\n"
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000168 << " .code { border-spacing:0px; width:100%; }\n"
169 << " .code { font-family: \"Andale Mono\", fixed; font-size:10pt }\n"
170 << " .code { line-height: 1.2em }\n"
171 << " .num { width:2.5em; padding-right:2ex; background-color:#eeeeee }\n"
172 << " .num { text-align:right; font-size: smaller }\n"
Ted Kremenek0f1b67b2008-03-19 23:55:53 +0000173 << " .num { color:#444444 }\n"
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000174 << " .line { padding-left: 1ex; border-left: 3px solid #ccc }\n"
175 << " .line { white-space: pre }\n"
Ted Kremenekeeacc522008-03-19 22:44:21 +0000176 << " .msg { background-color:#ff8000; color:#000000 }\n"
Ted Kremenek9d4e6592008-03-24 23:38:32 +0000177 << " .msg { -webkit-box-shadow:1px 1px 7px #000 }\n"
178 << " .msg { -webkit-border-radius:5px }\n"
Ted Kremenekeeacc522008-03-19 22:44:21 +0000179 << " .msg { border: solid 1px #944a00 }\n"
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000180 << " .msg { font-family:Helvetica, sans-serif; font-size: smaller }\n"
Ted Kremenekdb87ca02008-03-19 22:06:25 +0000181 << " .msg { font-weight: bold }\n"
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000182 << " .msg { float:left }\n"
Ted Kremenekeeacc522008-03-19 22:44:21 +0000183 << " .msg { padding:0.5em 1ex 0.5em 1ex }\n"
184 << " .msg { margin-top:10px; margin-bottom:10px }\n"
Ted Kremenek9d4e6592008-03-24 23:38:32 +0000185 << " .mrange { background-color:#ffcc66 }\n"
186 << " .mrange { border-bottom: 1px solid #ff8000 }\n"
Ted Kremeneke8e019e2008-03-19 21:59:05 +0000187 << "</style>\n</head>\n<body>";
Ted Kremenekad0a2032008-03-19 06:14:37 +0000188
189 R.InsertStrBefore(StartLoc, os.str());
190 }
191
192 // Generate footer
193
194 {
195 std::ostringstream os;
196
197 os << "</body></html>\n";
198 R.InsertStrAfter(EndLoc, os.str());
199 }
200}
201
202