summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--ui-shared.c6
-rw-r--r--ui-shared.h2
-rw-r--r--ui-tree.c8
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
13const char cgit_doctype[] = 13const 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
17static char *http_date(time_t t) 17static 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
30void cgit_print_error(char *msg) 30void 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
37char *cgit_rooturl() 37char *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
45char *cgit_repourl(const char *reponame) 45char *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
54char *cgit_fileurl(const char *reponame, const char *pagename, 54char *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
74char *cgit_pageurl(const char *reponame, const char *pagename, 74char *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
80const char *cgit_repobasename(const char *reponame) 80const 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
105char *cgit_currurl() 105char *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
117static void site_url(char *page, char *search, int ofs) 117static 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
144static void site_link(char *page, char *name, char *title, char *class, 144static 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
165void cgit_index_link(char *name, char *title, char *class, char *pattern, 165void 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
171static char *repolink(char *title, char *class, char *page, char *head, 171static 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 = "&amp;"; 213 delim = "&amp;";
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 = "&amp;"; 219 delim = "&amp;";
220 } 220 }
221 return fmt("%s", delim); 221 return fmt("%s", delim);
222} 222}
223 223
224static void reporevlink(char *page, char *name, char *title, char *class, 224static 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
240void cgit_tree_link(char *name, char *title, char *class, char *head, 240void 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
246void 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
246void cgit_log_link(char *name, char *title, char *class, char *head, 252void 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
277void cgit_commit_link(char *name, char *title, char *class, char *head, 283void 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
289void cgit_refs_link(char *name, char *title, char *class, char *head, 295void 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
295void cgit_snapshot_link(char *name, char *title, char *class, char *head, 301void 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
301void cgit_diff_link(char *name, char *title, char *class, char *head, 307void 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 = "&amp;"; 317 delim = "&amp;";
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
323void cgit_patch_link(char *name, char *title, char *class, char *head, 329void 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
329void cgit_object_link(struct object *obj) 335void 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
356void cgit_print_date(time_t secs, char *format, int local_time) 362void 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
371void cgit_print_age(time_t t, time_t max_relative, char *format) 377void 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
414void cgit_print_http_headers(struct cgit_context *ctx) 420void 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
431void cgit_print_docstart(struct cgit_context *ctx) 437void 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
454void cgit_print_docend() 460void 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
468int print_branch_option(const char *refname, const unsigned char *sha1, 474int 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
476int print_archive_ref(const char *refname, const unsigned char *sha1, 482int 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&amp;path=%s", sha1_to_hex(fileid), 513 fmt("id=%s&amp;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
515void add_hidden_formfields(int incl_head, int incl_search, char *page) 521void 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
543char *hc(struct cgit_cmd *cmd, const char *page) 549char *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
548void cgit_print_pageheader(struct cgit_context *ctx) 554void 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
648void cgit_print_filemode(unsigned short mode) 654void 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
663void cgit_print_snapshot_links(const char *repo, const char *head, 669void 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
4extern char *cgit_repourl(const char *reponame); 4extern char *cgit_repourl(const char *reponame);
5extern char *cgit_fileurl(const char *reponame, const char *pagename, 5extern char *cgit_fileurl(const char *reponame, const char *pagename,
6 const char *filename, const char *query); 6 const char *filename, const char *query);
7extern char *cgit_pageurl(const char *reponame, const char *pagename, 7extern char *cgit_pageurl(const char *reponame, const char *pagename,
8 const char *query); 8 const char *query);
9 9
10extern void cgit_index_link(char *name, char *title, char *class, 10extern void cgit_index_link(char *name, char *title, char *class,
11 char *pattern, int ofs); 11 char *pattern, int ofs);
12extern void cgit_tree_link(char *name, char *title, char *class, char *head, 12extern void cgit_tree_link(char *name, char *title, char *class, char *head,
13 char *rev, char *path); 13 char *rev, char *path);
14extern void cgit_plain_link(char *name, char *title, char *class, char *head,
15 char *rev, char *path);
14extern void cgit_log_link(char *name, char *title, char *class, char *head, 16extern 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);
17extern void cgit_commit_link(char *name, char *title, char *class, char *head, 19extern void cgit_commit_link(char *name, char *title, char *class, char *head,
18 char *rev); 20 char *rev);
19extern void cgit_patch_link(char *name, char *title, char *class, char *head, 21extern void cgit_patch_link(char *name, char *title, char *class, char *head,
20 char *rev); 22 char *rev);
21extern void cgit_refs_link(char *name, char *title, char *class, char *head, 23extern void cgit_refs_link(char *name, char *title, char *class, char *head,
22 char *rev, char *path); 24 char *rev, char *path);
23extern void cgit_snapshot_link(char *name, char *title, char *class, 25extern void cgit_snapshot_link(char *name, char *title, char *class,
24 char *head, char *rev, char *archivename); 26 char *head, char *rev, char *archivename);
25extern void cgit_diff_link(char *name, char *title, char *class, char *head, 27extern 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);
27extern void cgit_object_link(struct object *obj); 29extern void cgit_object_link(struct object *obj);
28 30
29extern void cgit_print_error(char *msg); 31extern void cgit_print_error(char *msg);
30extern void cgit_print_date(time_t secs, char *format, int local_time); 32extern void cgit_print_date(time_t secs, char *format, int local_time);
31extern void cgit_print_age(time_t t, time_t max_relative, char *format); 33extern void cgit_print_age(time_t t, time_t max_relative, char *format);
32extern void cgit_print_http_headers(struct cgit_context *ctx); 34extern void cgit_print_http_headers(struct cgit_context *ctx);
33extern void cgit_print_docstart(struct cgit_context *ctx); 35extern void cgit_print_docstart(struct cgit_context *ctx);
34extern void cgit_print_docend(); 36extern void cgit_print_docend();
35extern void cgit_print_pageheader(struct cgit_context *ctx); 37extern void cgit_print_pageheader(struct cgit_context *ctx);
36extern void cgit_print_filemode(unsigned short mode); 38extern void cgit_print_filemode(unsigned short mode);
37extern void cgit_print_snapshot_links(const char *repo, const char *head, 39extern 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 */
diff --git a/ui-tree.c b/ui-tree.c
index 9a837e2..79332fc 100644
--- a/ui-tree.c
+++ b/ui-tree.c
@@ -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
13char *curr_rev; 13char *curr_rev;
14char *match_path; 14char *match_path;
15int header = 0; 15int header = 0;
16 16
17static void print_object(const unsigned char *sha1, char *path) 17static 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
64static int ls_item(const unsigned char *sha1, const char *base, int baselen, 64static 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
115static void ls_head() 115static 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
127static void ls_tail() 127static 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
135static void ls_tree(const unsigned char *sha1, char *path) 135static 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
152static int walk_tree(const unsigned char *sha1, const char *base, int baselen, 152static 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&amp;path=%s", curr_rev, buffer)); 164 fmt("h=%s&amp;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 */
191void cgit_print_tree(const char *rev, char *path) 191void 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}