author | Lars Hjemli <hjemli@gmail.com> | 2008-08-06 09:07:13 (UTC) |
---|---|---|
committer | Lars Hjemli <hjemli@gmail.com> | 2008-08-06 09:21:30 (UTC) |
commit | 65b7b876aaaf50fc15060533359d6561f4f1819a (patch) (unidiff) | |
tree | c5cfe73456cf31afb13bcb12c5331fa711f89d71 | |
parent | e5da4bca54574522b28f88cab0dc8ebad9e35a73 (diff) | |
download | cgit-65b7b876aaaf50fc15060533359d6561f4f1819a.zip cgit-65b7b876aaaf50fc15060533359d6561f4f1819a.tar.gz cgit-65b7b876aaaf50fc15060533359d6561f4f1819a.tar.bz2 |
ui-tree: link to plain view instead of blob view
The urls for plain view makes it possible to download blobs without knowing
their SHA1, but the function needs to be promoted and the link from tree
view seems like a perfect fit.
PS: Although hidden, the blob view still is nice for direct blob access so
there's no point in removing it.
Signed-off-by: Lars Hjemli <hjemli@gmail.com>
-rw-r--r-- | ui-shared.c | 6 | ||||
-rw-r--r-- | ui-shared.h | 2 | ||||
-rw-r--r-- | ui-tree.c | 8 |
3 files changed, 12 insertions, 4 deletions
diff --git a/ui-shared.c b/ui-shared.c index 4408969..a2e0dd2 100644 --- a/ui-shared.c +++ b/ui-shared.c | |||
@@ -1,678 +1,684 @@ | |||
1 | /* ui-shared.c: common web output functions | 1 | /* ui-shared.c: common web output functions |
2 | * | 2 | * |
3 | * Copyright (C) 2006 Lars Hjemli | 3 | * Copyright (C) 2006 Lars Hjemli |
4 | * | 4 | * |
5 | * Licensed under GNU General Public License v2 | 5 | * Licensed under GNU General Public License v2 |
6 | * (see COPYING for full license text) | 6 | * (see COPYING for full license text) |
7 | */ | 7 | */ |
8 | 8 | ||
9 | #include "cgit.h" | 9 | #include "cgit.h" |
10 | #include "cmd.h" | 10 | #include "cmd.h" |
11 | #include "html.h" | 11 | #include "html.h" |
12 | 12 | ||
13 | const char cgit_doctype[] = | 13 | const char cgit_doctype[] = |
14 | "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\"\n" | 14 | "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\"\n" |
15 | " \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n"; | 15 | " \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n"; |
16 | 16 | ||
17 | static char *http_date(time_t t) | 17 | static char *http_date(time_t t) |
18 | { | 18 | { |
19 | static char day[][4] = | 19 | static char day[][4] = |
20 | {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; | 20 | {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; |
21 | static char month[][4] = | 21 | static char month[][4] = |
22 | {"Jan", "Feb", "Mar", "Apr", "May", "Jun", | 22 | {"Jan", "Feb", "Mar", "Apr", "May", "Jun", |
23 | "Jul", "Aug", "Sep", "Oct", "Now", "Dec"}; | 23 | "Jul", "Aug", "Sep", "Oct", "Now", "Dec"}; |
24 | struct tm *tm = gmtime(&t); | 24 | struct tm *tm = gmtime(&t); |
25 | return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday], | 25 | return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday], |
26 | tm->tm_mday, month[tm->tm_mon], 1900+tm->tm_year, | 26 | tm->tm_mday, month[tm->tm_mon], 1900+tm->tm_year, |
27 | tm->tm_hour, tm->tm_min, tm->tm_sec); | 27 | tm->tm_hour, tm->tm_min, tm->tm_sec); |
28 | } | 28 | } |
29 | 29 | ||
30 | void cgit_print_error(char *msg) | 30 | void cgit_print_error(char *msg) |
31 | { | 31 | { |
32 | html("<div class='error'>"); | 32 | html("<div class='error'>"); |
33 | html_txt(msg); | 33 | html_txt(msg); |
34 | html("</div>\n"); | 34 | html("</div>\n"); |
35 | } | 35 | } |
36 | 36 | ||
37 | char *cgit_rooturl() | 37 | char *cgit_rooturl() |
38 | { | 38 | { |
39 | if (ctx.cfg.virtual_root) | 39 | if (ctx.cfg.virtual_root) |
40 | return fmt("%s/", ctx.cfg.virtual_root); | 40 | return fmt("%s/", ctx.cfg.virtual_root); |
41 | else | 41 | else |
42 | return ctx.cfg.script_name; | 42 | return ctx.cfg.script_name; |
43 | } | 43 | } |
44 | 44 | ||
45 | char *cgit_repourl(const char *reponame) | 45 | char *cgit_repourl(const char *reponame) |
46 | { | 46 | { |
47 | if (ctx.cfg.virtual_root) { | 47 | if (ctx.cfg.virtual_root) { |
48 | return fmt("%s/%s/", ctx.cfg.virtual_root, reponame); | 48 | return fmt("%s/%s/", ctx.cfg.virtual_root, reponame); |
49 | } else { | 49 | } else { |
50 | return fmt("?r=%s", reponame); | 50 | return fmt("?r=%s", reponame); |
51 | } | 51 | } |
52 | } | 52 | } |
53 | 53 | ||
54 | char *cgit_fileurl(const char *reponame, const char *pagename, | 54 | char *cgit_fileurl(const char *reponame, const char *pagename, |
55 | const char *filename, const char *query) | 55 | const char *filename, const char *query) |
56 | { | 56 | { |
57 | char *tmp; | 57 | char *tmp; |
58 | char *delim; | 58 | char *delim; |
59 | 59 | ||
60 | if (ctx.cfg.virtual_root) { | 60 | if (ctx.cfg.virtual_root) { |
61 | tmp = fmt("%s/%s/%s/%s", ctx.cfg.virtual_root, reponame, | 61 | tmp = fmt("%s/%s/%s/%s", ctx.cfg.virtual_root, reponame, |
62 | pagename, (filename ? filename:"")); | 62 | pagename, (filename ? filename:"")); |
63 | delim = "?"; | 63 | delim = "?"; |
64 | } else { | 64 | } else { |
65 | tmp = fmt("?url=%s/%s/%s", reponame, pagename, | 65 | tmp = fmt("?url=%s/%s/%s", reponame, pagename, |
66 | (filename ? filename : "")); | 66 | (filename ? filename : "")); |
67 | delim = "&"; | 67 | delim = "&"; |
68 | } | 68 | } |
69 | if (query) | 69 | if (query) |
70 | tmp = fmt("%s%s%s", tmp, delim, query); | 70 | tmp = fmt("%s%s%s", tmp, delim, query); |
71 | return tmp; | 71 | return tmp; |
72 | } | 72 | } |
73 | 73 | ||
74 | char *cgit_pageurl(const char *reponame, const char *pagename, | 74 | char *cgit_pageurl(const char *reponame, const char *pagename, |
75 | const char *query) | 75 | const char *query) |
76 | { | 76 | { |
77 | return cgit_fileurl(reponame,pagename,0,query); | 77 | return cgit_fileurl(reponame,pagename,0,query); |
78 | } | 78 | } |
79 | 79 | ||
80 | const char *cgit_repobasename(const char *reponame) | 80 | const char *cgit_repobasename(const char *reponame) |
81 | { | 81 | { |
82 | /* I assume we don't need to store more than one repo basename */ | 82 | /* I assume we don't need to store more than one repo basename */ |
83 | static char rvbuf[1024]; | 83 | static char rvbuf[1024]; |
84 | int p; | 84 | int p; |
85 | const char *rv; | 85 | const char *rv; |
86 | strncpy(rvbuf,reponame,sizeof(rvbuf)); | 86 | strncpy(rvbuf,reponame,sizeof(rvbuf)); |
87 | if(rvbuf[sizeof(rvbuf)-1]) | 87 | if(rvbuf[sizeof(rvbuf)-1]) |
88 | die("cgit_repobasename: truncated repository name '%s'", reponame); | 88 | die("cgit_repobasename: truncated repository name '%s'", reponame); |
89 | p = strlen(rvbuf)-1; | 89 | p = strlen(rvbuf)-1; |
90 | /* strip trailing slashes */ | 90 | /* strip trailing slashes */ |
91 | while(p && rvbuf[p]=='/') rvbuf[p--]=0; | 91 | while(p && rvbuf[p]=='/') rvbuf[p--]=0; |
92 | /* strip trailing .git */ | 92 | /* strip trailing .git */ |
93 | if(p>=3 && !strncmp(&rvbuf[p-3],".git",4)) { | 93 | if(p>=3 && !strncmp(&rvbuf[p-3],".git",4)) { |
94 | p -= 3; rvbuf[p--] = 0; | 94 | p -= 3; rvbuf[p--] = 0; |
95 | } | 95 | } |
96 | /* strip more trailing slashes if any */ | 96 | /* strip more trailing slashes if any */ |
97 | while( p && rvbuf[p]=='/') rvbuf[p--]=0; | 97 | while( p && rvbuf[p]=='/') rvbuf[p--]=0; |
98 | /* find last slash in the remaining string */ | 98 | /* find last slash in the remaining string */ |
99 | rv = strrchr(rvbuf,'/'); | 99 | rv = strrchr(rvbuf,'/'); |
100 | if(rv) | 100 | if(rv) |
101 | return ++rv; | 101 | return ++rv; |
102 | return rvbuf; | 102 | return rvbuf; |
103 | } | 103 | } |
104 | 104 | ||
105 | char *cgit_currurl() | 105 | char *cgit_currurl() |
106 | { | 106 | { |
107 | if (!ctx.cfg.virtual_root) | 107 | if (!ctx.cfg.virtual_root) |
108 | return ctx.cfg.script_name; | 108 | return ctx.cfg.script_name; |
109 | else if (ctx.qry.page) | 109 | else if (ctx.qry.page) |
110 | return fmt("%s/%s/%s/", ctx.cfg.virtual_root, ctx.qry.repo, ctx.qry.page); | 110 | return fmt("%s/%s/%s/", ctx.cfg.virtual_root, ctx.qry.repo, ctx.qry.page); |
111 | else if (ctx.qry.repo) | 111 | else if (ctx.qry.repo) |
112 | return fmt("%s/%s/", ctx.cfg.virtual_root, ctx.qry.repo); | 112 | return fmt("%s/%s/", ctx.cfg.virtual_root, ctx.qry.repo); |
113 | else | 113 | else |
114 | return fmt("%s/", ctx.cfg.virtual_root); | 114 | return fmt("%s/", ctx.cfg.virtual_root); |
115 | } | 115 | } |
116 | 116 | ||
117 | static void site_url(char *page, char *search, int ofs) | 117 | static void site_url(char *page, char *search, int ofs) |
118 | { | 118 | { |
119 | char *delim = "?"; | 119 | char *delim = "?"; |
120 | 120 | ||
121 | if (ctx.cfg.virtual_root) { | 121 | if (ctx.cfg.virtual_root) { |
122 | html_attr(ctx.cfg.virtual_root); | 122 | html_attr(ctx.cfg.virtual_root); |
123 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') | 123 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') |
124 | html("/"); | 124 | html("/"); |
125 | } else | 125 | } else |
126 | html(ctx.cfg.script_name); | 126 | html(ctx.cfg.script_name); |
127 | 127 | ||
128 | if (page) { | 128 | if (page) { |
129 | htmlf("?p=%s", page); | 129 | htmlf("?p=%s", page); |
130 | delim = "&"; | 130 | delim = "&"; |
131 | } | 131 | } |
132 | if (search) { | 132 | if (search) { |
133 | html(delim); | 133 | html(delim); |
134 | html("q="); | 134 | html("q="); |
135 | html_attr(search); | 135 | html_attr(search); |
136 | delim = "&"; | 136 | delim = "&"; |
137 | } | 137 | } |
138 | if (ofs) { | 138 | if (ofs) { |
139 | html(delim); | 139 | html(delim); |
140 | htmlf("ofs=%d", ofs); | 140 | htmlf("ofs=%d", ofs); |
141 | } | 141 | } |
142 | } | 142 | } |
143 | 143 | ||
144 | static void site_link(char *page, char *name, char *title, char *class, | 144 | static void site_link(char *page, char *name, char *title, char *class, |
145 | char *search, int ofs) | 145 | char *search, int ofs) |
146 | { | 146 | { |
147 | html("<a"); | 147 | html("<a"); |
148 | if (title) { | 148 | if (title) { |
149 | html(" title='"); | 149 | html(" title='"); |
150 | html_attr(title); | 150 | html_attr(title); |
151 | html("'"); | 151 | html("'"); |
152 | } | 152 | } |
153 | if (class) { | 153 | if (class) { |
154 | html(" class='"); | 154 | html(" class='"); |
155 | html_attr(class); | 155 | html_attr(class); |
156 | html("'"); | 156 | html("'"); |
157 | } | 157 | } |
158 | html(" href='"); | 158 | html(" href='"); |
159 | site_url(page, search, ofs); | 159 | site_url(page, search, ofs); |
160 | html("'>"); | 160 | html("'>"); |
161 | html_txt(name); | 161 | html_txt(name); |
162 | html("</a>"); | 162 | html("</a>"); |
163 | } | 163 | } |
164 | 164 | ||
165 | void cgit_index_link(char *name, char *title, char *class, char *pattern, | 165 | void cgit_index_link(char *name, char *title, char *class, char *pattern, |
166 | int ofs) | 166 | int ofs) |
167 | { | 167 | { |
168 | site_link(NULL, name, title, class, pattern, ofs); | 168 | site_link(NULL, name, title, class, pattern, ofs); |
169 | } | 169 | } |
170 | 170 | ||
171 | static char *repolink(char *title, char *class, char *page, char *head, | 171 | static char *repolink(char *title, char *class, char *page, char *head, |
172 | char *path) | 172 | char *path) |
173 | { | 173 | { |
174 | char *delim = "?"; | 174 | char *delim = "?"; |
175 | 175 | ||
176 | html("<a"); | 176 | html("<a"); |
177 | if (title) { | 177 | if (title) { |
178 | html(" title='"); | 178 | html(" title='"); |
179 | html_attr(title); | 179 | html_attr(title); |
180 | html("'"); | 180 | html("'"); |
181 | } | 181 | } |
182 | if (class) { | 182 | if (class) { |
183 | html(" class='"); | 183 | html(" class='"); |
184 | html_attr(class); | 184 | html_attr(class); |
185 | html("'"); | 185 | html("'"); |
186 | } | 186 | } |
187 | html(" href='"); | 187 | html(" href='"); |
188 | if (ctx.cfg.virtual_root) { | 188 | if (ctx.cfg.virtual_root) { |
189 | html_attr(ctx.cfg.virtual_root); | 189 | html_attr(ctx.cfg.virtual_root); |
190 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') | 190 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') |
191 | html("/"); | 191 | html("/"); |
192 | html_attr(ctx.repo->url); | 192 | html_attr(ctx.repo->url); |
193 | if (ctx.repo->url[strlen(ctx.repo->url) - 1] != '/') | 193 | if (ctx.repo->url[strlen(ctx.repo->url) - 1] != '/') |
194 | html("/"); | 194 | html("/"); |
195 | if (page) { | 195 | if (page) { |
196 | html(page); | 196 | html(page); |
197 | html("/"); | 197 | html("/"); |
198 | if (path) | 198 | if (path) |
199 | html_attr(path); | 199 | html_attr(path); |
200 | } | 200 | } |
201 | } else { | 201 | } else { |
202 | html(ctx.cfg.script_name); | 202 | html(ctx.cfg.script_name); |
203 | html("?url="); | 203 | html("?url="); |
204 | html_attr(ctx.repo->url); | 204 | html_attr(ctx.repo->url); |
205 | if (ctx.repo->url[strlen(ctx.repo->url) - 1] != '/') | 205 | if (ctx.repo->url[strlen(ctx.repo->url) - 1] != '/') |
206 | html("/"); | 206 | html("/"); |
207 | if (page) { | 207 | if (page) { |
208 | html(page); | 208 | html(page); |
209 | html("/"); | 209 | html("/"); |
210 | if (path) | 210 | if (path) |
211 | html_attr(path); | 211 | html_attr(path); |
212 | } | 212 | } |
213 | delim = "&"; | 213 | delim = "&"; |
214 | } | 214 | } |
215 | if (head && strcmp(head, ctx.repo->defbranch)) { | 215 | if (head && strcmp(head, ctx.repo->defbranch)) { |
216 | html(delim); | 216 | html(delim); |
217 | html("h="); | 217 | html("h="); |
218 | html_attr(head); | 218 | html_attr(head); |
219 | delim = "&"; | 219 | delim = "&"; |
220 | } | 220 | } |
221 | return fmt("%s", delim); | 221 | return fmt("%s", delim); |
222 | } | 222 | } |
223 | 223 | ||
224 | static void reporevlink(char *page, char *name, char *title, char *class, | 224 | static void reporevlink(char *page, char *name, char *title, char *class, |
225 | char *head, char *rev, char *path) | 225 | char *head, char *rev, char *path) |
226 | { | 226 | { |
227 | char *delim; | 227 | char *delim; |
228 | 228 | ||
229 | delim = repolink(title, class, page, head, path); | 229 | delim = repolink(title, class, page, head, path); |
230 | if (rev && strcmp(rev, ctx.qry.head)) { | 230 | if (rev && strcmp(rev, ctx.qry.head)) { |
231 | html(delim); | 231 | html(delim); |
232 | html("id="); | 232 | html("id="); |
233 | html_attr(rev); | 233 | html_attr(rev); |
234 | } | 234 | } |
235 | html("'>"); | 235 | html("'>"); |
236 | html_txt(name); | 236 | html_txt(name); |
237 | html("</a>"); | 237 | html("</a>"); |
238 | } | 238 | } |
239 | 239 | ||
240 | void cgit_tree_link(char *name, char *title, char *class, char *head, | 240 | void cgit_tree_link(char *name, char *title, char *class, char *head, |
241 | char *rev, char *path) | 241 | char *rev, char *path) |
242 | { | 242 | { |
243 | reporevlink("tree", name, title, class, head, rev, path); | 243 | reporevlink("tree", name, title, class, head, rev, path); |
244 | } | 244 | } |
245 | 245 | ||
246 | void cgit_plain_link(char *name, char *title, char *class, char *head, | ||
247 | char *rev, char *path) | ||
248 | { | ||
249 | reporevlink("plain", name, title, class, head, rev, path); | ||
250 | } | ||
251 | |||
246 | void cgit_log_link(char *name, char *title, char *class, char *head, | 252 | void cgit_log_link(char *name, char *title, char *class, char *head, |
247 | char *rev, char *path, int ofs, char *grep, char *pattern) | 253 | char *rev, char *path, int ofs, char *grep, char *pattern) |
248 | { | 254 | { |
249 | char *delim; | 255 | char *delim; |
250 | 256 | ||
251 | delim = repolink(title, class, "log", head, path); | 257 | delim = repolink(title, class, "log", head, path); |
252 | if (rev && strcmp(rev, ctx.qry.head)) { | 258 | if (rev && strcmp(rev, ctx.qry.head)) { |
253 | html(delim); | 259 | html(delim); |
254 | html("id="); | 260 | html("id="); |
255 | html_attr(rev); | 261 | html_attr(rev); |
256 | delim = "&"; | 262 | delim = "&"; |
257 | } | 263 | } |
258 | if (grep && pattern) { | 264 | if (grep && pattern) { |
259 | html(delim); | 265 | html(delim); |
260 | html("qt="); | 266 | html("qt="); |
261 | html_attr(grep); | 267 | html_attr(grep); |
262 | delim = "&"; | 268 | delim = "&"; |
263 | html(delim); | 269 | html(delim); |
264 | html("q="); | 270 | html("q="); |
265 | html_attr(pattern); | 271 | html_attr(pattern); |
266 | } | 272 | } |
267 | if (ofs > 0) { | 273 | if (ofs > 0) { |
268 | html(delim); | 274 | html(delim); |
269 | html("ofs="); | 275 | html("ofs="); |
270 | htmlf("%d", ofs); | 276 | htmlf("%d", ofs); |
271 | } | 277 | } |
272 | html("'>"); | 278 | html("'>"); |
273 | html_txt(name); | 279 | html_txt(name); |
274 | html("</a>"); | 280 | html("</a>"); |
275 | } | 281 | } |
276 | 282 | ||
277 | void cgit_commit_link(char *name, char *title, char *class, char *head, | 283 | void cgit_commit_link(char *name, char *title, char *class, char *head, |
278 | char *rev) | 284 | char *rev) |
279 | { | 285 | { |
280 | if (strlen(name) > ctx.cfg.max_msg_len && ctx.cfg.max_msg_len >= 15) { | 286 | if (strlen(name) > ctx.cfg.max_msg_len && ctx.cfg.max_msg_len >= 15) { |
281 | name[ctx.cfg.max_msg_len] = '\0'; | 287 | name[ctx.cfg.max_msg_len] = '\0'; |
282 | name[ctx.cfg.max_msg_len - 1] = '.'; | 288 | name[ctx.cfg.max_msg_len - 1] = '.'; |
283 | name[ctx.cfg.max_msg_len - 2] = '.'; | 289 | name[ctx.cfg.max_msg_len - 2] = '.'; |
284 | name[ctx.cfg.max_msg_len - 3] = '.'; | 290 | name[ctx.cfg.max_msg_len - 3] = '.'; |
285 | } | 291 | } |
286 | reporevlink("commit", name, title, class, head, rev, NULL); | 292 | reporevlink("commit", name, title, class, head, rev, NULL); |
287 | } | 293 | } |
288 | 294 | ||
289 | void cgit_refs_link(char *name, char *title, char *class, char *head, | 295 | void cgit_refs_link(char *name, char *title, char *class, char *head, |
290 | char *rev, char *path) | 296 | char *rev, char *path) |
291 | { | 297 | { |
292 | reporevlink("refs", name, title, class, head, rev, path); | 298 | reporevlink("refs", name, title, class, head, rev, path); |
293 | } | 299 | } |
294 | 300 | ||
295 | void cgit_snapshot_link(char *name, char *title, char *class, char *head, | 301 | void cgit_snapshot_link(char *name, char *title, char *class, char *head, |
296 | char *rev, char *archivename) | 302 | char *rev, char *archivename) |
297 | { | 303 | { |
298 | reporevlink("snapshot", name, title, class, head, rev, archivename); | 304 | reporevlink("snapshot", name, title, class, head, rev, archivename); |
299 | } | 305 | } |
300 | 306 | ||
301 | void cgit_diff_link(char *name, char *title, char *class, char *head, | 307 | void cgit_diff_link(char *name, char *title, char *class, char *head, |
302 | char *new_rev, char *old_rev, char *path) | 308 | char *new_rev, char *old_rev, char *path) |
303 | { | 309 | { |
304 | char *delim; | 310 | char *delim; |
305 | 311 | ||
306 | delim = repolink(title, class, "diff", head, path); | 312 | delim = repolink(title, class, "diff", head, path); |
307 | if (new_rev && strcmp(new_rev, ctx.qry.head)) { | 313 | if (new_rev && strcmp(new_rev, ctx.qry.head)) { |
308 | html(delim); | 314 | html(delim); |
309 | html("id="); | 315 | html("id="); |
310 | html_attr(new_rev); | 316 | html_attr(new_rev); |
311 | delim = "&"; | 317 | delim = "&"; |
312 | } | 318 | } |
313 | if (old_rev) { | 319 | if (old_rev) { |
314 | html(delim); | 320 | html(delim); |
315 | html("id2="); | 321 | html("id2="); |
316 | html_attr(old_rev); | 322 | html_attr(old_rev); |
317 | } | 323 | } |
318 | html("'>"); | 324 | html("'>"); |
319 | html_txt(name); | 325 | html_txt(name); |
320 | html("</a>"); | 326 | html("</a>"); |
321 | } | 327 | } |
322 | 328 | ||
323 | void cgit_patch_link(char *name, char *title, char *class, char *head, | 329 | void cgit_patch_link(char *name, char *title, char *class, char *head, |
324 | char *rev) | 330 | char *rev) |
325 | { | 331 | { |
326 | reporevlink("patch", name, title, class, head, rev, NULL); | 332 | reporevlink("patch", name, title, class, head, rev, NULL); |
327 | } | 333 | } |
328 | 334 | ||
329 | void cgit_object_link(struct object *obj) | 335 | void cgit_object_link(struct object *obj) |
330 | { | 336 | { |
331 | char *page, *arg, *url; | 337 | char *page, *arg, *url; |
332 | 338 | ||
333 | if (obj->type == OBJ_COMMIT) { | 339 | if (obj->type == OBJ_COMMIT) { |
334 | cgit_commit_link(fmt("commit %s", sha1_to_hex(obj->sha1)), NULL, NULL, | 340 | cgit_commit_link(fmt("commit %s", sha1_to_hex(obj->sha1)), NULL, NULL, |
335 | ctx.qry.head, sha1_to_hex(obj->sha1)); | 341 | ctx.qry.head, sha1_to_hex(obj->sha1)); |
336 | return; | 342 | return; |
337 | } else if (obj->type == OBJ_TREE) { | 343 | } else if (obj->type == OBJ_TREE) { |
338 | page = "tree"; | 344 | page = "tree"; |
339 | arg = "id"; | 345 | arg = "id"; |
340 | } else if (obj->type == OBJ_TAG) { | 346 | } else if (obj->type == OBJ_TAG) { |
341 | page = "tag"; | 347 | page = "tag"; |
342 | arg = "id"; | 348 | arg = "id"; |
343 | } else { | 349 | } else { |
344 | page = "blob"; | 350 | page = "blob"; |
345 | arg = "id"; | 351 | arg = "id"; |
346 | } | 352 | } |
347 | 353 | ||
348 | url = cgit_pageurl(ctx.qry.repo, page, | 354 | url = cgit_pageurl(ctx.qry.repo, page, |
349 | fmt("%s=%s", arg, sha1_to_hex(obj->sha1))); | 355 | fmt("%s=%s", arg, sha1_to_hex(obj->sha1))); |
350 | html_link_open(url, NULL, NULL); | 356 | html_link_open(url, NULL, NULL); |
351 | htmlf("%s %s", typename(obj->type), | 357 | htmlf("%s %s", typename(obj->type), |
352 | sha1_to_hex(obj->sha1)); | 358 | sha1_to_hex(obj->sha1)); |
353 | html_link_close(); | 359 | html_link_close(); |
354 | } | 360 | } |
355 | 361 | ||
356 | void cgit_print_date(time_t secs, char *format, int local_time) | 362 | void cgit_print_date(time_t secs, char *format, int local_time) |
357 | { | 363 | { |
358 | char buf[64]; | 364 | char buf[64]; |
359 | struct tm *time; | 365 | struct tm *time; |
360 | 366 | ||
361 | if (!secs) | 367 | if (!secs) |
362 | return; | 368 | return; |
363 | if(local_time) | 369 | if(local_time) |
364 | time = localtime(&secs); | 370 | time = localtime(&secs); |
365 | else | 371 | else |
366 | time = gmtime(&secs); | 372 | time = gmtime(&secs); |
367 | strftime(buf, sizeof(buf)-1, format, time); | 373 | strftime(buf, sizeof(buf)-1, format, time); |
368 | html_txt(buf); | 374 | html_txt(buf); |
369 | } | 375 | } |
370 | 376 | ||
371 | void cgit_print_age(time_t t, time_t max_relative, char *format) | 377 | void cgit_print_age(time_t t, time_t max_relative, char *format) |
372 | { | 378 | { |
373 | time_t now, secs; | 379 | time_t now, secs; |
374 | 380 | ||
375 | if (!t) | 381 | if (!t) |
376 | return; | 382 | return; |
377 | time(&now); | 383 | time(&now); |
378 | secs = now - t; | 384 | secs = now - t; |
379 | 385 | ||
380 | if (secs > max_relative && max_relative >= 0) { | 386 | if (secs > max_relative && max_relative >= 0) { |
381 | cgit_print_date(t, format, ctx.cfg.local_time); | 387 | cgit_print_date(t, format, ctx.cfg.local_time); |
382 | return; | 388 | return; |
383 | } | 389 | } |
384 | 390 | ||
385 | if (secs < TM_HOUR * 2) { | 391 | if (secs < TM_HOUR * 2) { |
386 | htmlf("<span class='age-mins'>%.0f min.</span>", | 392 | htmlf("<span class='age-mins'>%.0f min.</span>", |
387 | secs * 1.0 / TM_MIN); | 393 | secs * 1.0 / TM_MIN); |
388 | return; | 394 | return; |
389 | } | 395 | } |
390 | if (secs < TM_DAY * 2) { | 396 | if (secs < TM_DAY * 2) { |
391 | htmlf("<span class='age-hours'>%.0f hours</span>", | 397 | htmlf("<span class='age-hours'>%.0f hours</span>", |
392 | secs * 1.0 / TM_HOUR); | 398 | secs * 1.0 / TM_HOUR); |
393 | return; | 399 | return; |
394 | } | 400 | } |
395 | if (secs < TM_WEEK * 2) { | 401 | if (secs < TM_WEEK * 2) { |
396 | htmlf("<span class='age-days'>%.0f days</span>", | 402 | htmlf("<span class='age-days'>%.0f days</span>", |
397 | secs * 1.0 / TM_DAY); | 403 | secs * 1.0 / TM_DAY); |
398 | return; | 404 | return; |
399 | } | 405 | } |
400 | if (secs < TM_MONTH * 2) { | 406 | if (secs < TM_MONTH * 2) { |
401 | htmlf("<span class='age-weeks'>%.0f weeks</span>", | 407 | htmlf("<span class='age-weeks'>%.0f weeks</span>", |
402 | secs * 1.0 / TM_WEEK); | 408 | secs * 1.0 / TM_WEEK); |
403 | return; | 409 | return; |
404 | } | 410 | } |
405 | if (secs < TM_YEAR * 2) { | 411 | if (secs < TM_YEAR * 2) { |
406 | htmlf("<span class='age-months'>%.0f months</span>", | 412 | htmlf("<span class='age-months'>%.0f months</span>", |
407 | secs * 1.0 / TM_MONTH); | 413 | secs * 1.0 / TM_MONTH); |
408 | return; | 414 | return; |
409 | } | 415 | } |
410 | htmlf("<span class='age-years'>%.0f years</span>", | 416 | htmlf("<span class='age-years'>%.0f years</span>", |
411 | secs * 1.0 / TM_YEAR); | 417 | secs * 1.0 / TM_YEAR); |
412 | } | 418 | } |
413 | 419 | ||
414 | void cgit_print_http_headers(struct cgit_context *ctx) | 420 | void cgit_print_http_headers(struct cgit_context *ctx) |
415 | { | 421 | { |
416 | if (ctx->page.mimetype && ctx->page.charset) | 422 | if (ctx->page.mimetype && ctx->page.charset) |
417 | htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype, | 423 | htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype, |
418 | ctx->page.charset); | 424 | ctx->page.charset); |
419 | else if (ctx->page.mimetype) | 425 | else if (ctx->page.mimetype) |
420 | htmlf("Content-Type: %s\n", ctx->page.mimetype); | 426 | htmlf("Content-Type: %s\n", ctx->page.mimetype); |
421 | if (ctx->page.size) | 427 | if (ctx->page.size) |
422 | htmlf("Content-Length: %ld\n", ctx->page.size); | 428 | htmlf("Content-Length: %ld\n", ctx->page.size); |
423 | if (ctx->page.filename) | 429 | if (ctx->page.filename) |
424 | htmlf("Content-Disposition: inline; filename=\"%s\"\n", | 430 | htmlf("Content-Disposition: inline; filename=\"%s\"\n", |
425 | ctx->page.filename); | 431 | ctx->page.filename); |
426 | htmlf("Last-Modified: %s\n", http_date(ctx->page.modified)); | 432 | htmlf("Last-Modified: %s\n", http_date(ctx->page.modified)); |
427 | htmlf("Expires: %s\n", http_date(ctx->page.expires)); | 433 | htmlf("Expires: %s\n", http_date(ctx->page.expires)); |
428 | html("\n"); | 434 | html("\n"); |
429 | } | 435 | } |
430 | 436 | ||
431 | void cgit_print_docstart(struct cgit_context *ctx) | 437 | void cgit_print_docstart(struct cgit_context *ctx) |
432 | { | 438 | { |
433 | html(cgit_doctype); | 439 | html(cgit_doctype); |
434 | html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n"); | 440 | html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n"); |
435 | html("<head>\n"); | 441 | html("<head>\n"); |
436 | html("<title>"); | 442 | html("<title>"); |
437 | html_txt(ctx->page.title); | 443 | html_txt(ctx->page.title); |
438 | html("</title>\n"); | 444 | html("</title>\n"); |
439 | htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version); | 445 | htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version); |
440 | if (ctx->cfg.robots && *ctx->cfg.robots) | 446 | if (ctx->cfg.robots && *ctx->cfg.robots) |
441 | htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots); | 447 | htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots); |
442 | html("<link rel='stylesheet' type='text/css' href='"); | 448 | html("<link rel='stylesheet' type='text/css' href='"); |
443 | html_attr(ctx->cfg.css); | 449 | html_attr(ctx->cfg.css); |
444 | html("'/>\n"); | 450 | html("'/>\n"); |
445 | if (ctx->cfg.favicon) { | 451 | if (ctx->cfg.favicon) { |
446 | html("<link rel='shortcut icon' href='"); | 452 | html("<link rel='shortcut icon' href='"); |
447 | html_attr(ctx->cfg.favicon); | 453 | html_attr(ctx->cfg.favicon); |
448 | html("'/>\n"); | 454 | html("'/>\n"); |
449 | } | 455 | } |
450 | html("</head>\n"); | 456 | html("</head>\n"); |
451 | html("<body>\n"); | 457 | html("<body>\n"); |
452 | } | 458 | } |
453 | 459 | ||
454 | void cgit_print_docend() | 460 | void cgit_print_docend() |
455 | { | 461 | { |
456 | html("</div>"); | 462 | html("</div>"); |
457 | if (ctx.cfg.footer) | 463 | if (ctx.cfg.footer) |
458 | html_include(ctx.cfg.footer); | 464 | html_include(ctx.cfg.footer); |
459 | else { | 465 | else { |
460 | html("<div class='footer'>generated "); | 466 | html("<div class='footer'>generated "); |
461 | cgit_print_date(time(NULL), FMT_LONGDATE, ctx.cfg.local_time); | 467 | cgit_print_date(time(NULL), FMT_LONGDATE, ctx.cfg.local_time); |
462 | htmlf(" by cgit %s", cgit_version); | 468 | htmlf(" by cgit %s", cgit_version); |
463 | html("</div>\n"); | 469 | html("</div>\n"); |
464 | } | 470 | } |
465 | html("</body>\n</html>\n"); | 471 | html("</body>\n</html>\n"); |
466 | } | 472 | } |
467 | 473 | ||
468 | int print_branch_option(const char *refname, const unsigned char *sha1, | 474 | int print_branch_option(const char *refname, const unsigned char *sha1, |
469 | int flags, void *cb_data) | 475 | int flags, void *cb_data) |
470 | { | 476 | { |
471 | char *name = (char *)refname; | 477 | char *name = (char *)refname; |
472 | html_option(name, name, ctx.qry.head); | 478 | html_option(name, name, ctx.qry.head); |
473 | return 0; | 479 | return 0; |
474 | } | 480 | } |
475 | 481 | ||
476 | int print_archive_ref(const char *refname, const unsigned char *sha1, | 482 | int print_archive_ref(const char *refname, const unsigned char *sha1, |
477 | int flags, void *cb_data) | 483 | int flags, void *cb_data) |
478 | { | 484 | { |
479 | struct tag *tag; | 485 | struct tag *tag; |
480 | struct taginfo *info; | 486 | struct taginfo *info; |
481 | struct object *obj; | 487 | struct object *obj; |
482 | char buf[256], *url; | 488 | char buf[256], *url; |
483 | unsigned char fileid[20]; | 489 | unsigned char fileid[20]; |
484 | int *header = (int *)cb_data; | 490 | int *header = (int *)cb_data; |
485 | 491 | ||
486 | if (prefixcmp(refname, "refs/archives")) | 492 | if (prefixcmp(refname, "refs/archives")) |
487 | return 0; | 493 | return 0; |
488 | strncpy(buf, refname+14, sizeof(buf)); | 494 | strncpy(buf, refname+14, sizeof(buf)); |
489 | obj = parse_object(sha1); | 495 | obj = parse_object(sha1); |
490 | if (!obj) | 496 | if (!obj) |
491 | return 1; | 497 | return 1; |
492 | if (obj->type == OBJ_TAG) { | 498 | if (obj->type == OBJ_TAG) { |
493 | tag = lookup_tag(sha1); | 499 | tag = lookup_tag(sha1); |
494 | if (!tag || parse_tag(tag) || !(info = cgit_parse_tag(tag))) | 500 | if (!tag || parse_tag(tag) || !(info = cgit_parse_tag(tag))) |
495 | return 0; | 501 | return 0; |
496 | hashcpy(fileid, tag->tagged->sha1); | 502 | hashcpy(fileid, tag->tagged->sha1); |
497 | } else if (obj->type != OBJ_BLOB) { | 503 | } else if (obj->type != OBJ_BLOB) { |
498 | return 0; | 504 | return 0; |
499 | } else { | 505 | } else { |
500 | hashcpy(fileid, sha1); | 506 | hashcpy(fileid, sha1); |
501 | } | 507 | } |
502 | if (!*header) { | 508 | if (!*header) { |
503 | html("<h1>download</h1>\n"); | 509 | html("<h1>download</h1>\n"); |
504 | *header = 1; | 510 | *header = 1; |
505 | } | 511 | } |
506 | url = cgit_pageurl(ctx.qry.repo, "blob", | 512 | url = cgit_pageurl(ctx.qry.repo, "blob", |
507 | fmt("id=%s&path=%s", sha1_to_hex(fileid), | 513 | fmt("id=%s&path=%s", sha1_to_hex(fileid), |
508 | buf)); | 514 | buf)); |
509 | html_link_open(url, NULL, "menu"); | 515 | html_link_open(url, NULL, "menu"); |
510 | html_txt(strlpart(buf, 20)); | 516 | html_txt(strlpart(buf, 20)); |
511 | html_link_close(); | 517 | html_link_close(); |
512 | return 0; | 518 | return 0; |
513 | } | 519 | } |
514 | 520 | ||
515 | void add_hidden_formfields(int incl_head, int incl_search, char *page) | 521 | void add_hidden_formfields(int incl_head, int incl_search, char *page) |
516 | { | 522 | { |
517 | char *url; | 523 | char *url; |
518 | 524 | ||
519 | if (!ctx.cfg.virtual_root) { | 525 | if (!ctx.cfg.virtual_root) { |
520 | url = fmt("%s/%s", ctx.qry.repo, page); | 526 | url = fmt("%s/%s", ctx.qry.repo, page); |
521 | if (ctx.qry.path) | 527 | if (ctx.qry.path) |
522 | url = fmt("%s/%s", url, ctx.qry.path); | 528 | url = fmt("%s/%s", url, ctx.qry.path); |
523 | html_hidden("url", url); | 529 | html_hidden("url", url); |
524 | } | 530 | } |
525 | 531 | ||
526 | if (incl_head && ctx.qry.head && ctx.repo->defbranch && | 532 | if (incl_head && ctx.qry.head && ctx.repo->defbranch && |
527 | strcmp(ctx.qry.head, ctx.repo->defbranch)) | 533 | strcmp(ctx.qry.head, ctx.repo->defbranch)) |
528 | html_hidden("h", ctx.qry.head); | 534 | html_hidden("h", ctx.qry.head); |
529 | 535 | ||
530 | if (ctx.qry.sha1) | 536 | if (ctx.qry.sha1) |
531 | html_hidden("id", ctx.qry.sha1); | 537 | html_hidden("id", ctx.qry.sha1); |
532 | if (ctx.qry.sha2) | 538 | if (ctx.qry.sha2) |
533 | html_hidden("id2", ctx.qry.sha2); | 539 | html_hidden("id2", ctx.qry.sha2); |
534 | 540 | ||
535 | if (incl_search) { | 541 | if (incl_search) { |
536 | if (ctx.qry.grep) | 542 | if (ctx.qry.grep) |
537 | html_hidden("qt", ctx.qry.grep); | 543 | html_hidden("qt", ctx.qry.grep); |
538 | if (ctx.qry.search) | 544 | if (ctx.qry.search) |
539 | html_hidden("q", ctx.qry.search); | 545 | html_hidden("q", ctx.qry.search); |
540 | } | 546 | } |
541 | } | 547 | } |
542 | 548 | ||
543 | char *hc(struct cgit_cmd *cmd, const char *page) | 549 | char *hc(struct cgit_cmd *cmd, const char *page) |
544 | { | 550 | { |
545 | return (strcmp(cmd->name, page) ? NULL : "active"); | 551 | return (strcmp(cmd->name, page) ? NULL : "active"); |
546 | } | 552 | } |
547 | 553 | ||
548 | void cgit_print_pageheader(struct cgit_context *ctx) | 554 | void cgit_print_pageheader(struct cgit_context *ctx) |
549 | { | 555 | { |
550 | struct cgit_cmd *cmd = cgit_get_cmd(ctx); | 556 | struct cgit_cmd *cmd = cgit_get_cmd(ctx); |
551 | 557 | ||
552 | html("<table id='header'>\n"); | 558 | html("<table id='header'>\n"); |
553 | html("<tr>\n"); | 559 | html("<tr>\n"); |
554 | html("<td class='logo' rowspan='2'><a href='"); | 560 | html("<td class='logo' rowspan='2'><a href='"); |
555 | if (ctx->cfg.logo_link) | 561 | if (ctx->cfg.logo_link) |
556 | html_attr(ctx->cfg.logo_link); | 562 | html_attr(ctx->cfg.logo_link); |
557 | else | 563 | else |
558 | html_attr(cgit_rooturl()); | 564 | html_attr(cgit_rooturl()); |
559 | html("'><img src='"); | 565 | html("'><img src='"); |
560 | html_attr(ctx->cfg.logo); | 566 | html_attr(ctx->cfg.logo); |
561 | html("' alt='cgit logo'/></a></td>\n"); | 567 | html("' alt='cgit logo'/></a></td>\n"); |
562 | 568 | ||
563 | html("<td class='main'>"); | 569 | html("<td class='main'>"); |
564 | if (ctx->repo) { | 570 | if (ctx->repo) { |
565 | cgit_index_link("index", NULL, NULL, NULL, 0); | 571 | cgit_index_link("index", NULL, NULL, NULL, 0); |
566 | html(" : "); | 572 | html(" : "); |
567 | reporevlink(NULL, ctx->repo->name, NULL, hc(cmd, "summary"), | 573 | reporevlink(NULL, ctx->repo->name, NULL, hc(cmd, "summary"), |
568 | ctx->qry.head, NULL, NULL); | 574 | ctx->qry.head, NULL, NULL); |
569 | html("</td><td class='form'>"); | 575 | html("</td><td class='form'>"); |
570 | html("<form method='get' action=''>\n"); | 576 | html("<form method='get' action=''>\n"); |
571 | add_hidden_formfields(0, 1, ctx->qry.page); | 577 | add_hidden_formfields(0, 1, ctx->qry.page); |
572 | html("<select name='h' onchange='this.form.submit();'>\n"); | 578 | html("<select name='h' onchange='this.form.submit();'>\n"); |
573 | for_each_branch_ref(print_branch_option, ctx->qry.head); | 579 | for_each_branch_ref(print_branch_option, ctx->qry.head); |
574 | html("</select> "); | 580 | html("</select> "); |
575 | html("<input type='submit' name='' value='switch'/>"); | 581 | html("<input type='submit' name='' value='switch'/>"); |
576 | html("</form>"); | 582 | html("</form>"); |
577 | } else | 583 | } else |
578 | html_txt(ctx->cfg.root_title); | 584 | html_txt(ctx->cfg.root_title); |
579 | html("</td></tr>\n"); | 585 | html("</td></tr>\n"); |
580 | 586 | ||
581 | html("<tr><td class='sub'>"); | 587 | html("<tr><td class='sub'>"); |
582 | if (ctx->repo) { | 588 | if (ctx->repo) { |
583 | html_txt(ctx->repo->desc); | 589 | html_txt(ctx->repo->desc); |
584 | html("</td><td class='sub right'>"); | 590 | html("</td><td class='sub right'>"); |
585 | html_txt(ctx->repo->owner); | 591 | html_txt(ctx->repo->owner); |
586 | } else { | 592 | } else { |
587 | if (ctx->cfg.root_desc) | 593 | if (ctx->cfg.root_desc) |
588 | html_txt(ctx->cfg.root_desc); | 594 | html_txt(ctx->cfg.root_desc); |
589 | else if (ctx->cfg.index_info) | 595 | else if (ctx->cfg.index_info) |
590 | html_include(ctx->cfg.index_info); | 596 | html_include(ctx->cfg.index_info); |
591 | } | 597 | } |
592 | html("</td></tr></table>\n"); | 598 | html("</td></tr></table>\n"); |
593 | 599 | ||
594 | html("<table class='tabs'><tr><td>\n"); | 600 | html("<table class='tabs'><tr><td>\n"); |
595 | if (ctx->repo) { | 601 | if (ctx->repo) { |
596 | reporevlink(NULL, "summary", NULL, hc(cmd, "summary"), | 602 | reporevlink(NULL, "summary", NULL, hc(cmd, "summary"), |
597 | ctx->qry.head, NULL, NULL); | 603 | ctx->qry.head, NULL, NULL); |
598 | cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, | 604 | cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, |
599 | ctx->qry.sha1, NULL); | 605 | ctx->qry.sha1, NULL); |
600 | cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, | 606 | cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, |
601 | NULL, NULL, 0, NULL, NULL); | 607 | NULL, NULL, 0, NULL, NULL); |
602 | cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, | 608 | cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, |
603 | ctx->qry.sha1, NULL); | 609 | ctx->qry.sha1, NULL); |
604 | cgit_commit_link("commit", NULL, hc(cmd, "commit"), | 610 | cgit_commit_link("commit", NULL, hc(cmd, "commit"), |
605 | ctx->qry.head, ctx->qry.sha1); | 611 | ctx->qry.head, ctx->qry.sha1); |
606 | cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, | 612 | cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, |
607 | ctx->qry.sha1, ctx->qry.sha2, NULL); | 613 | ctx->qry.sha1, ctx->qry.sha2, NULL); |
608 | if (ctx->repo->readme) | 614 | if (ctx->repo->readme) |
609 | reporevlink("about", "about", NULL, | 615 | reporevlink("about", "about", NULL, |
610 | hc(cmd, "about"), ctx->qry.head, NULL, | 616 | hc(cmd, "about"), ctx->qry.head, NULL, |
611 | NULL); | 617 | NULL); |
612 | html("</td><td class='form'>"); | 618 | html("</td><td class='form'>"); |
613 | html("<form class='right' method='get' action='"); | 619 | html("<form class='right' method='get' action='"); |
614 | if (ctx->cfg.virtual_root) | 620 | if (ctx->cfg.virtual_root) |
615 | html_attr(cgit_fileurl(ctx->qry.repo, "log", | 621 | html_attr(cgit_fileurl(ctx->qry.repo, "log", |
616 | ctx->qry.path, NULL)); | 622 | ctx->qry.path, NULL)); |
617 | html("'>\n"); | 623 | html("'>\n"); |
618 | add_hidden_formfields(1, 0, "log"); | 624 | add_hidden_formfields(1, 0, "log"); |
619 | html("<select name='qt'>\n"); | 625 | html("<select name='qt'>\n"); |
620 | html_option("grep", "log msg", ctx->qry.grep); | 626 | html_option("grep", "log msg", ctx->qry.grep); |
621 | html_option("author", "author", ctx->qry.grep); | 627 | html_option("author", "author", ctx->qry.grep); |
622 | html_option("committer", "committer", ctx->qry.grep); | 628 | html_option("committer", "committer", ctx->qry.grep); |
623 | html("</select>\n"); | 629 | html("</select>\n"); |
624 | html("<input class='txt' type='text' size='10' name='q' value='"); | 630 | html("<input class='txt' type='text' size='10' name='q' value='"); |
625 | html_attr(ctx->qry.search); | 631 | html_attr(ctx->qry.search); |
626 | html("'/>\n"); | 632 | html("'/>\n"); |
627 | html("<input type='submit' value='search'/>\n"); | 633 | html("<input type='submit' value='search'/>\n"); |
628 | html("</form>\n"); | 634 | html("</form>\n"); |
629 | } else { | 635 | } else { |
630 | site_link(NULL, "index", NULL, hc(cmd, "repolist"), NULL, 0); | 636 | site_link(NULL, "index", NULL, hc(cmd, "repolist"), NULL, 0); |
631 | if (ctx->cfg.root_readme) | 637 | if (ctx->cfg.root_readme) |
632 | site_link("about", "about", NULL, hc(cmd, "about"), | 638 | site_link("about", "about", NULL, hc(cmd, "about"), |
633 | NULL, 0); | 639 | NULL, 0); |
634 | html("</td><td class='form'>"); | 640 | html("</td><td class='form'>"); |
635 | html("<form method='get' action='"); | 641 | html("<form method='get' action='"); |
636 | html_attr(cgit_rooturl()); | 642 | html_attr(cgit_rooturl()); |
637 | html("'>\n"); | 643 | html("'>\n"); |
638 | html("<input type='text' name='q' size='10' value='"); | 644 | html("<input type='text' name='q' size='10' value='"); |
639 | html_attr(ctx->qry.search); | 645 | html_attr(ctx->qry.search); |
640 | html("'/>\n"); | 646 | html("'/>\n"); |
641 | html("<input type='submit' value='search'/>\n"); | 647 | html("<input type='submit' value='search'/>\n"); |
642 | html("</form>"); | 648 | html("</form>"); |
643 | } | 649 | } |
644 | html("</td></tr></table>\n"); | 650 | html("</td></tr></table>\n"); |
645 | html("<div class='content'>"); | 651 | html("<div class='content'>"); |
646 | } | 652 | } |
647 | 653 | ||
648 | void cgit_print_filemode(unsigned short mode) | 654 | void cgit_print_filemode(unsigned short mode) |
649 | { | 655 | { |
650 | if (S_ISDIR(mode)) | 656 | if (S_ISDIR(mode)) |
651 | html("d"); | 657 | html("d"); |
652 | else if (S_ISLNK(mode)) | 658 | else if (S_ISLNK(mode)) |
653 | html("l"); | 659 | html("l"); |
654 | else if (S_ISGITLINK(mode)) | 660 | else if (S_ISGITLINK(mode)) |
655 | html("m"); | 661 | html("m"); |
656 | else | 662 | else |
657 | html("-"); | 663 | html("-"); |
658 | html_fileperm(mode >> 6); | 664 | html_fileperm(mode >> 6); |
659 | html_fileperm(mode >> 3); | 665 | html_fileperm(mode >> 3); |
660 | html_fileperm(mode); | 666 | html_fileperm(mode); |
661 | } | 667 | } |
662 | 668 | ||
663 | void cgit_print_snapshot_links(const char *repo, const char *head, | 669 | void cgit_print_snapshot_links(const char *repo, const char *head, |
664 | const char *hex, int snapshots) | 670 | const char *hex, int snapshots) |
665 | { | 671 | { |
666 | const struct cgit_snapshot_format* f; | 672 | const struct cgit_snapshot_format* f; |
667 | char *filename; | 673 | char *filename; |
668 | 674 | ||
669 | for (f = cgit_snapshot_formats; f->suffix; f++) { | 675 | for (f = cgit_snapshot_formats; f->suffix; f++) { |
670 | if (!(snapshots & f->bit)) | 676 | if (!(snapshots & f->bit)) |
671 | continue; | 677 | continue; |
672 | filename = fmt("%s-%s%s", cgit_repobasename(repo), hex, | 678 | filename = fmt("%s-%s%s", cgit_repobasename(repo), hex, |
673 | f->suffix); | 679 | f->suffix); |
674 | cgit_snapshot_link(filename, NULL, NULL, (char *)head, | 680 | cgit_snapshot_link(filename, NULL, NULL, (char *)head, |
675 | (char *)hex, filename); | 681 | (char *)hex, filename); |
676 | html("<br/>"); | 682 | html("<br/>"); |
677 | } | 683 | } |
678 | } | 684 | } |
diff --git a/ui-shared.h b/ui-shared.h index 07da4b4..c5ce056 100644 --- a/ui-shared.h +++ b/ui-shared.h | |||
@@ -1,40 +1,42 @@ | |||
1 | #ifndef UI_SHARED_H | 1 | #ifndef UI_SHARED_H |
2 | #define UI_SHARED_H | 2 | #define UI_SHARED_H |
3 | 3 | ||
4 | extern char *cgit_repourl(const char *reponame); | 4 | extern char *cgit_repourl(const char *reponame); |
5 | extern char *cgit_fileurl(const char *reponame, const char *pagename, | 5 | extern char *cgit_fileurl(const char *reponame, const char *pagename, |
6 | const char *filename, const char *query); | 6 | const char *filename, const char *query); |
7 | extern char *cgit_pageurl(const char *reponame, const char *pagename, | 7 | extern char *cgit_pageurl(const char *reponame, const char *pagename, |
8 | const char *query); | 8 | const char *query); |
9 | 9 | ||
10 | extern void cgit_index_link(char *name, char *title, char *class, | 10 | extern void cgit_index_link(char *name, char *title, char *class, |
11 | char *pattern, int ofs); | 11 | char *pattern, int ofs); |
12 | extern void cgit_tree_link(char *name, char *title, char *class, char *head, | 12 | extern void cgit_tree_link(char *name, char *title, char *class, char *head, |
13 | char *rev, char *path); | 13 | char *rev, char *path); |
14 | extern void cgit_plain_link(char *name, char *title, char *class, char *head, | ||
15 | char *rev, char *path); | ||
14 | extern void cgit_log_link(char *name, char *title, char *class, char *head, | 16 | extern void cgit_log_link(char *name, char *title, char *class, char *head, |
15 | char *rev, char *path, int ofs, char *grep, | 17 | char *rev, char *path, int ofs, char *grep, |
16 | char *pattern); | 18 | char *pattern); |
17 | extern void cgit_commit_link(char *name, char *title, char *class, char *head, | 19 | extern void cgit_commit_link(char *name, char *title, char *class, char *head, |
18 | char *rev); | 20 | char *rev); |
19 | extern void cgit_patch_link(char *name, char *title, char *class, char *head, | 21 | extern void cgit_patch_link(char *name, char *title, char *class, char *head, |
20 | char *rev); | 22 | char *rev); |
21 | extern void cgit_refs_link(char *name, char *title, char *class, char *head, | 23 | extern void cgit_refs_link(char *name, char *title, char *class, char *head, |
22 | char *rev, char *path); | 24 | char *rev, char *path); |
23 | extern void cgit_snapshot_link(char *name, char *title, char *class, | 25 | extern void cgit_snapshot_link(char *name, char *title, char *class, |
24 | char *head, char *rev, char *archivename); | 26 | char *head, char *rev, char *archivename); |
25 | extern void cgit_diff_link(char *name, char *title, char *class, char *head, | 27 | extern void cgit_diff_link(char *name, char *title, char *class, char *head, |
26 | char *new_rev, char *old_rev, char *path); | 28 | char *new_rev, char *old_rev, char *path); |
27 | extern void cgit_object_link(struct object *obj); | 29 | extern void cgit_object_link(struct object *obj); |
28 | 30 | ||
29 | extern void cgit_print_error(char *msg); | 31 | extern void cgit_print_error(char *msg); |
30 | extern void cgit_print_date(time_t secs, char *format, int local_time); | 32 | extern void cgit_print_date(time_t secs, char *format, int local_time); |
31 | extern void cgit_print_age(time_t t, time_t max_relative, char *format); | 33 | extern void cgit_print_age(time_t t, time_t max_relative, char *format); |
32 | extern void cgit_print_http_headers(struct cgit_context *ctx); | 34 | extern void cgit_print_http_headers(struct cgit_context *ctx); |
33 | extern void cgit_print_docstart(struct cgit_context *ctx); | 35 | extern void cgit_print_docstart(struct cgit_context *ctx); |
34 | extern void cgit_print_docend(); | 36 | extern void cgit_print_docend(); |
35 | extern void cgit_print_pageheader(struct cgit_context *ctx); | 37 | extern void cgit_print_pageheader(struct cgit_context *ctx); |
36 | extern void cgit_print_filemode(unsigned short mode); | 38 | extern void cgit_print_filemode(unsigned short mode); |
37 | extern void cgit_print_snapshot_links(const char *repo, const char *head, | 39 | extern void cgit_print_snapshot_links(const char *repo, const char *head, |
38 | const char *hex, int snapshots); | 40 | const char *hex, int snapshots); |
39 | 41 | ||
40 | #endif /* UI_SHARED_H */ | 42 | #endif /* UI_SHARED_H */ |
@@ -1,223 +1,223 @@ | |||
1 | /* ui-tree.c: functions for tree output | 1 | /* ui-tree.c: functions for tree output |
2 | * | 2 | * |
3 | * Copyright (C) 2006 Lars Hjemli | 3 | * Copyright (C) 2006 Lars Hjemli |
4 | * | 4 | * |
5 | * Licensed under GNU General Public License v2 | 5 | * Licensed under GNU General Public License v2 |
6 | * (see COPYING for full license text) | 6 | * (see COPYING for full license text) |
7 | */ | 7 | */ |
8 | 8 | ||
9 | #include "cgit.h" | 9 | #include "cgit.h" |
10 | #include "html.h" | 10 | #include "html.h" |
11 | #include "ui-shared.h" | 11 | #include "ui-shared.h" |
12 | 12 | ||
13 | char *curr_rev; | 13 | char *curr_rev; |
14 | char *match_path; | 14 | char *match_path; |
15 | int header = 0; | 15 | int header = 0; |
16 | 16 | ||
17 | static void print_object(const unsigned char *sha1, char *path) | 17 | static void print_object(const unsigned char *sha1, char *path) |
18 | { | 18 | { |
19 | enum object_type type; | 19 | enum object_type type; |
20 | char *buf; | 20 | char *buf; |
21 | unsigned long size, lineno, start, idx; | 21 | unsigned long size, lineno, start, idx; |
22 | const char *linefmt = "<tr><td class='no'><a id='n%1$d' name='n%1$d' href='#n%1$d'>%1$d</a></td><td class='txt'>"; | 22 | const char *linefmt = "<tr><td class='no'><a id='n%1$d' name='n%1$d' href='#n%1$d'>%1$d</a></td><td class='txt'>"; |
23 | 23 | ||
24 | type = sha1_object_info(sha1, &size); | 24 | type = sha1_object_info(sha1, &size); |
25 | if (type == OBJ_BAD) { | 25 | if (type == OBJ_BAD) { |
26 | cgit_print_error(fmt("Bad object name: %s", | 26 | cgit_print_error(fmt("Bad object name: %s", |
27 | sha1_to_hex(sha1))); | 27 | sha1_to_hex(sha1))); |
28 | return; | 28 | return; |
29 | } | 29 | } |
30 | 30 | ||
31 | buf = read_sha1_file(sha1, &type, &size); | 31 | buf = read_sha1_file(sha1, &type, &size); |
32 | if (!buf) { | 32 | if (!buf) { |
33 | cgit_print_error(fmt("Error reading object %s", | 33 | cgit_print_error(fmt("Error reading object %s", |
34 | sha1_to_hex(sha1))); | 34 | sha1_to_hex(sha1))); |
35 | return; | 35 | return; |
36 | } | 36 | } |
37 | 37 | ||
38 | html(" blob: <a href='"); | 38 | html(" ("); |
39 | html_attr(cgit_pageurl(ctx.qry.repo, "blob", | 39 | cgit_plain_link("plain", NULL, NULL, ctx.qry.head, |
40 | fmt("id=%s&path=%s", sha1_to_hex(sha1), path))); | 40 | curr_rev, path); |
41 | htmlf("'>%s</a>",sha1_to_hex(sha1)); | 41 | htmlf(")<br/>blob: %s", sha1_to_hex(sha1)); |
42 | 42 | ||
43 | html("<table summary='blob content' class='blob'>\n"); | 43 | html("<table summary='blob content' class='blob'>\n"); |
44 | idx = 0; | 44 | idx = 0; |
45 | start = 0; | 45 | start = 0; |
46 | lineno = 0; | 46 | lineno = 0; |
47 | while(idx < size) { | 47 | while(idx < size) { |
48 | if (buf[idx] == '\n') { | 48 | if (buf[idx] == '\n') { |
49 | buf[idx] = '\0'; | 49 | buf[idx] = '\0'; |
50 | htmlf(linefmt, ++lineno); | 50 | htmlf(linefmt, ++lineno); |
51 | html_txt(buf + start); | 51 | html_txt(buf + start); |
52 | html("</td></tr>\n"); | 52 | html("</td></tr>\n"); |
53 | start = idx + 1; | 53 | start = idx + 1; |
54 | } | 54 | } |
55 | idx++; | 55 | idx++; |
56 | } | 56 | } |
57 | htmlf(linefmt, ++lineno); | 57 | htmlf(linefmt, ++lineno); |
58 | html_txt(buf + start); | 58 | html_txt(buf + start); |
59 | html("</td></tr>\n"); | 59 | html("</td></tr>\n"); |
60 | html("</table>\n"); | 60 | html("</table>\n"); |
61 | } | 61 | } |
62 | 62 | ||
63 | 63 | ||
64 | static int ls_item(const unsigned char *sha1, const char *base, int baselen, | 64 | static int ls_item(const unsigned char *sha1, const char *base, int baselen, |
65 | const char *pathname, unsigned int mode, int stage, | 65 | const char *pathname, unsigned int mode, int stage, |
66 | void *cbdata) | 66 | void *cbdata) |
67 | { | 67 | { |
68 | char *name; | 68 | char *name; |
69 | char *fullpath; | 69 | char *fullpath; |
70 | enum object_type type; | 70 | enum object_type type; |
71 | unsigned long size = 0; | 71 | unsigned long size = 0; |
72 | 72 | ||
73 | name = xstrdup(pathname); | 73 | name = xstrdup(pathname); |
74 | fullpath = fmt("%s%s%s", ctx.qry.path ? ctx.qry.path : "", | 74 | fullpath = fmt("%s%s%s", ctx.qry.path ? ctx.qry.path : "", |
75 | ctx.qry.path ? "/" : "", name); | 75 | ctx.qry.path ? "/" : "", name); |
76 | 76 | ||
77 | if (!S_ISGITLINK(mode)) { | 77 | if (!S_ISGITLINK(mode)) { |
78 | type = sha1_object_info(sha1, &size); | 78 | type = sha1_object_info(sha1, &size); |
79 | if (type == OBJ_BAD) { | 79 | if (type == OBJ_BAD) { |
80 | htmlf("<tr><td colspan='3'>Bad object: %s %s</td></tr>", | 80 | htmlf("<tr><td colspan='3'>Bad object: %s %s</td></tr>", |
81 | name, | 81 | name, |
82 | sha1_to_hex(sha1)); | 82 | sha1_to_hex(sha1)); |
83 | return 0; | 83 | return 0; |
84 | } | 84 | } |
85 | } | 85 | } |
86 | 86 | ||
87 | html("<tr><td class='ls-mode'>"); | 87 | html("<tr><td class='ls-mode'>"); |
88 | cgit_print_filemode(mode); | 88 | cgit_print_filemode(mode); |
89 | html("</td><td>"); | 89 | html("</td><td>"); |
90 | if (S_ISGITLINK(mode)) { | 90 | if (S_ISGITLINK(mode)) { |
91 | htmlf("<a class='ls-mod' href='"); | 91 | htmlf("<a class='ls-mod' href='"); |
92 | html_attr(fmt(ctx.repo->module_link, | 92 | html_attr(fmt(ctx.repo->module_link, |
93 | name, | 93 | name, |
94 | sha1_to_hex(sha1))); | 94 | sha1_to_hex(sha1))); |
95 | html("'>"); | 95 | html("'>"); |
96 | html_txt(name); | 96 | html_txt(name); |
97 | html("</a>"); | 97 | html("</a>"); |
98 | } else if (S_ISDIR(mode)) { | 98 | } else if (S_ISDIR(mode)) { |
99 | cgit_tree_link(name, NULL, "ls-dir", ctx.qry.head, | 99 | cgit_tree_link(name, NULL, "ls-dir", ctx.qry.head, |
100 | curr_rev, fullpath); | 100 | curr_rev, fullpath); |
101 | } else { | 101 | } else { |
102 | cgit_tree_link(name, NULL, "ls-blob", ctx.qry.head, | 102 | cgit_tree_link(name, NULL, "ls-blob", ctx.qry.head, |
103 | curr_rev, fullpath); | 103 | curr_rev, fullpath); |
104 | } | 104 | } |
105 | htmlf("</td><td class='ls-size'>%li</td>", size); | 105 | htmlf("</td><td class='ls-size'>%li</td>", size); |
106 | 106 | ||
107 | html("<td>"); | 107 | html("<td>"); |
108 | cgit_log_link("log", NULL, "button", ctx.qry.head, curr_rev, | 108 | cgit_log_link("log", NULL, "button", ctx.qry.head, curr_rev, |
109 | fullpath, 0, NULL, NULL); | 109 | fullpath, 0, NULL, NULL); |
110 | html("</td></tr>\n"); | 110 | html("</td></tr>\n"); |
111 | free(name); | 111 | free(name); |
112 | return 0; | 112 | return 0; |
113 | } | 113 | } |
114 | 114 | ||
115 | static void ls_head() | 115 | static void ls_head() |
116 | { | 116 | { |
117 | html("<table summary='tree listing' class='list'>\n"); | 117 | html("<table summary='tree listing' class='list'>\n"); |
118 | html("<tr class='nohover'>"); | 118 | html("<tr class='nohover'>"); |
119 | html("<th class='left'>Mode</th>"); | 119 | html("<th class='left'>Mode</th>"); |
120 | html("<th class='left'>Name</th>"); | 120 | html("<th class='left'>Name</th>"); |
121 | html("<th class='right'>Size</th>"); | 121 | html("<th class='right'>Size</th>"); |
122 | html("<th/>"); | 122 | html("<th/>"); |
123 | html("</tr>\n"); | 123 | html("</tr>\n"); |
124 | header = 1; | 124 | header = 1; |
125 | } | 125 | } |
126 | 126 | ||
127 | static void ls_tail() | 127 | static void ls_tail() |
128 | { | 128 | { |
129 | if (!header) | 129 | if (!header) |
130 | return; | 130 | return; |
131 | html("</table>\n"); | 131 | html("</table>\n"); |
132 | header = 0; | 132 | header = 0; |
133 | } | 133 | } |
134 | 134 | ||
135 | static void ls_tree(const unsigned char *sha1, char *path) | 135 | static void ls_tree(const unsigned char *sha1, char *path) |
136 | { | 136 | { |
137 | struct tree *tree; | 137 | struct tree *tree; |
138 | 138 | ||
139 | tree = parse_tree_indirect(sha1); | 139 | tree = parse_tree_indirect(sha1); |
140 | if (!tree) { | 140 | if (!tree) { |
141 | cgit_print_error(fmt("Not a tree object: %s", | 141 | cgit_print_error(fmt("Not a tree object: %s", |
142 | sha1_to_hex(sha1))); | 142 | sha1_to_hex(sha1))); |
143 | return; | 143 | return; |
144 | } | 144 | } |
145 | 145 | ||
146 | ls_head(); | 146 | ls_head(); |
147 | read_tree_recursive(tree, "", 0, 1, NULL, ls_item, NULL); | 147 | read_tree_recursive(tree, "", 0, 1, NULL, ls_item, NULL); |
148 | ls_tail(); | 148 | ls_tail(); |
149 | } | 149 | } |
150 | 150 | ||
151 | 151 | ||
152 | static int walk_tree(const unsigned char *sha1, const char *base, int baselen, | 152 | static int walk_tree(const unsigned char *sha1, const char *base, int baselen, |
153 | const char *pathname, unsigned mode, int stage, | 153 | const char *pathname, unsigned mode, int stage, |
154 | void *cbdata) | 154 | void *cbdata) |
155 | { | 155 | { |
156 | static int state; | 156 | static int state; |
157 | static char buffer[PATH_MAX]; | 157 | static char buffer[PATH_MAX]; |
158 | char *url; | 158 | char *url; |
159 | 159 | ||
160 | if (state == 0) { | 160 | if (state == 0) { |
161 | memcpy(buffer, base, baselen); | 161 | memcpy(buffer, base, baselen); |
162 | strcpy(buffer+baselen, pathname); | 162 | strcpy(buffer+baselen, pathname); |
163 | url = cgit_pageurl(ctx.qry.repo, "tree", | 163 | url = cgit_pageurl(ctx.qry.repo, "tree", |
164 | fmt("h=%s&path=%s", curr_rev, buffer)); | 164 | fmt("h=%s&path=%s", curr_rev, buffer)); |
165 | html("/"); | 165 | html("/"); |
166 | cgit_tree_link(xstrdup(pathname), NULL, NULL, ctx.qry.head, | 166 | cgit_tree_link(xstrdup(pathname), NULL, NULL, ctx.qry.head, |
167 | curr_rev, buffer); | 167 | curr_rev, buffer); |
168 | 168 | ||
169 | if (strcmp(match_path, buffer)) | 169 | if (strcmp(match_path, buffer)) |
170 | return READ_TREE_RECURSIVE; | 170 | return READ_TREE_RECURSIVE; |
171 | 171 | ||
172 | if (S_ISDIR(mode)) { | 172 | if (S_ISDIR(mode)) { |
173 | state = 1; | 173 | state = 1; |
174 | ls_head(); | 174 | ls_head(); |
175 | return READ_TREE_RECURSIVE; | 175 | return READ_TREE_RECURSIVE; |
176 | } else { | 176 | } else { |
177 | print_object(sha1, buffer); | 177 | print_object(sha1, buffer); |
178 | return 0; | 178 | return 0; |
179 | } | 179 | } |
180 | } | 180 | } |
181 | ls_item(sha1, base, baselen, pathname, mode, stage, NULL); | 181 | ls_item(sha1, base, baselen, pathname, mode, stage, NULL); |
182 | return 0; | 182 | return 0; |
183 | } | 183 | } |
184 | 184 | ||
185 | 185 | ||
186 | /* | 186 | /* |
187 | * Show a tree or a blob | 187 | * Show a tree or a blob |
188 | * rev: the commit pointing at the root tree object | 188 | * rev: the commit pointing at the root tree object |
189 | * path: path to tree or blob | 189 | * path: path to tree or blob |
190 | */ | 190 | */ |
191 | void cgit_print_tree(const char *rev, char *path) | 191 | void cgit_print_tree(const char *rev, char *path) |
192 | { | 192 | { |
193 | unsigned char sha1[20]; | 193 | unsigned char sha1[20]; |
194 | struct commit *commit; | 194 | struct commit *commit; |
195 | const char *paths[] = {path, NULL}; | 195 | const char *paths[] = {path, NULL}; |
196 | 196 | ||
197 | if (!rev) | 197 | if (!rev) |
198 | rev = ctx.qry.head; | 198 | rev = ctx.qry.head; |
199 | 199 | ||
200 | curr_rev = xstrdup(rev); | 200 | curr_rev = xstrdup(rev); |
201 | if (get_sha1(rev, sha1)) { | 201 | if (get_sha1(rev, sha1)) { |
202 | cgit_print_error(fmt("Invalid revision name: %s", rev)); | 202 | cgit_print_error(fmt("Invalid revision name: %s", rev)); |
203 | return; | 203 | return; |
204 | } | 204 | } |
205 | commit = lookup_commit_reference(sha1); | 205 | commit = lookup_commit_reference(sha1); |
206 | if (!commit || parse_commit(commit)) { | 206 | if (!commit || parse_commit(commit)) { |
207 | cgit_print_error(fmt("Invalid commit reference: %s", rev)); | 207 | cgit_print_error(fmt("Invalid commit reference: %s", rev)); |
208 | return; | 208 | return; |
209 | } | 209 | } |
210 | 210 | ||
211 | html("path: <a href='"); | 211 | html("path: <a href='"); |
212 | html_attr(cgit_pageurl(ctx.qry.repo, "tree", fmt("h=%s", rev))); | 212 | html_attr(cgit_pageurl(ctx.qry.repo, "tree", fmt("h=%s", rev))); |
213 | html("'>root</a>"); | 213 | html("'>root</a>"); |
214 | 214 | ||
215 | if (path == NULL) { | 215 | if (path == NULL) { |
216 | ls_tree(commit->tree->object.sha1, NULL); | 216 | ls_tree(commit->tree->object.sha1, NULL); |
217 | return; | 217 | return; |
218 | } | 218 | } |
219 | 219 | ||
220 | match_path = path; | 220 | match_path = path; |
221 | read_tree_recursive(commit->tree, NULL, 0, 0, paths, walk_tree, NULL); | 221 | read_tree_recursive(commit->tree, NULL, 0, 0, paths, walk_tree, NULL); |
222 | ls_tail(); | 222 | ls_tail(); |
223 | } | 223 | } |