-rw-r--r-- | cmd.c | 9 | ||||
-rw-r--r-- | ui-shared.c | 53 |
2 files changed, 59 insertions, 3 deletions
@@ -1,112 +1,121 @@ | |||
1 | /* cmd.c: the cgit command dispatcher | 1 | /* cmd.c: the cgit command dispatcher |
2 | * | 2 | * |
3 | * Copyright (C) 2008 Lars Hjemli | 3 | * Copyright (C) 2008 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 "ui-blob.h" | 11 | #include "ui-blob.h" |
12 | #include "ui-commit.h" | 12 | #include "ui-commit.h" |
13 | #include "ui-diff.h" | 13 | #include "ui-diff.h" |
14 | #include "ui-log.h" | 14 | #include "ui-log.h" |
15 | #include "ui-patch.h" | 15 | #include "ui-patch.h" |
16 | #include "ui-refs.h" | 16 | #include "ui-refs.h" |
17 | #include "ui-repolist.h" | 17 | #include "ui-repolist.h" |
18 | #include "ui-snapshot.h" | 18 | #include "ui-snapshot.h" |
19 | #include "ui-summary.h" | 19 | #include "ui-summary.h" |
20 | #include "ui-tag.h" | 20 | #include "ui-tag.h" |
21 | #include "ui-tree.h" | 21 | #include "ui-tree.h" |
22 | 22 | ||
23 | static void about_fn(struct cgit_context *ctx) | ||
24 | { | ||
25 | if (ctx->repo) | ||
26 | cgit_print_repo_readme(); | ||
27 | else | ||
28 | cgit_print_site_readme(); | ||
29 | } | ||
30 | |||
23 | static void blob_fn(struct cgit_context *ctx) | 31 | static void blob_fn(struct cgit_context *ctx) |
24 | { | 32 | { |
25 | cgit_print_blob(ctx->qry.sha1, ctx->qry.path); | 33 | cgit_print_blob(ctx->qry.sha1, ctx->qry.path); |
26 | } | 34 | } |
27 | 35 | ||
28 | static void commit_fn(struct cgit_context *ctx) | 36 | static void commit_fn(struct cgit_context *ctx) |
29 | { | 37 | { |
30 | cgit_print_commit(ctx->qry.sha1); | 38 | cgit_print_commit(ctx->qry.sha1); |
31 | } | 39 | } |
32 | 40 | ||
33 | static void diff_fn(struct cgit_context *ctx) | 41 | static void diff_fn(struct cgit_context *ctx) |
34 | { | 42 | { |
35 | cgit_print_diff(ctx->qry.sha1, ctx->qry.sha2, ctx->qry.path); | 43 | cgit_print_diff(ctx->qry.sha1, ctx->qry.sha2, ctx->qry.path); |
36 | } | 44 | } |
37 | 45 | ||
38 | static void repolist_fn(struct cgit_context *ctx) | 46 | static void repolist_fn(struct cgit_context *ctx) |
39 | { | 47 | { |
40 | cgit_print_repolist(); | 48 | cgit_print_repolist(); |
41 | } | 49 | } |
42 | 50 | ||
43 | static void log_fn(struct cgit_context *ctx) | 51 | static void log_fn(struct cgit_context *ctx) |
44 | { | 52 | { |
45 | cgit_print_log(ctx->qry.sha1, ctx->qry.ofs, ctx->cfg.max_commit_count, | 53 | cgit_print_log(ctx->qry.sha1, ctx->qry.ofs, ctx->cfg.max_commit_count, |
46 | ctx->qry.grep, ctx->qry.search, ctx->qry.path, 1); | 54 | ctx->qry.grep, ctx->qry.search, ctx->qry.path, 1); |
47 | } | 55 | } |
48 | 56 | ||
49 | static void patch_fn(struct cgit_context *ctx) | 57 | static void patch_fn(struct cgit_context *ctx) |
50 | { | 58 | { |
51 | cgit_print_patch(ctx->qry.sha1); | 59 | cgit_print_patch(ctx->qry.sha1); |
52 | } | 60 | } |
53 | 61 | ||
54 | static void refs_fn(struct cgit_context *ctx) | 62 | static void refs_fn(struct cgit_context *ctx) |
55 | { | 63 | { |
56 | cgit_print_refs(); | 64 | cgit_print_refs(); |
57 | } | 65 | } |
58 | 66 | ||
59 | static void snapshot_fn(struct cgit_context *ctx) | 67 | static void snapshot_fn(struct cgit_context *ctx) |
60 | { | 68 | { |
61 | cgit_print_snapshot(ctx->qry.head, ctx->qry.sha1, | 69 | cgit_print_snapshot(ctx->qry.head, ctx->qry.sha1, |
62 | cgit_repobasename(ctx->repo->url), ctx->qry.path, | 70 | cgit_repobasename(ctx->repo->url), ctx->qry.path, |
63 | ctx->repo->snapshots); | 71 | ctx->repo->snapshots); |
64 | } | 72 | } |
65 | 73 | ||
66 | static void summary_fn(struct cgit_context *ctx) | 74 | static void summary_fn(struct cgit_context *ctx) |
67 | { | 75 | { |
68 | cgit_print_summary(); | 76 | cgit_print_summary(); |
69 | } | 77 | } |
70 | 78 | ||
71 | static void tag_fn(struct cgit_context *ctx) | 79 | static void tag_fn(struct cgit_context *ctx) |
72 | { | 80 | { |
73 | cgit_print_tag(ctx->qry.sha1); | 81 | cgit_print_tag(ctx->qry.sha1); |
74 | } | 82 | } |
75 | 83 | ||
76 | static void tree_fn(struct cgit_context *ctx) | 84 | static void tree_fn(struct cgit_context *ctx) |
77 | { | 85 | { |
78 | cgit_print_tree(ctx->qry.sha1, ctx->qry.path); | 86 | cgit_print_tree(ctx->qry.sha1, ctx->qry.path); |
79 | } | 87 | } |
80 | 88 | ||
81 | #define def_cmd(name, want_repo, want_layout) \ | 89 | #define def_cmd(name, want_repo, want_layout) \ |
82 | {#name, name##_fn, want_repo, want_layout} | 90 | {#name, name##_fn, want_repo, want_layout} |
83 | 91 | ||
84 | struct cgit_cmd *cgit_get_cmd(struct cgit_context *ctx) | 92 | struct cgit_cmd *cgit_get_cmd(struct cgit_context *ctx) |
85 | { | 93 | { |
86 | static struct cgit_cmd cmds[] = { | 94 | static struct cgit_cmd cmds[] = { |
95 | def_cmd(about, 0, 1), | ||
87 | def_cmd(blob, 1, 0), | 96 | def_cmd(blob, 1, 0), |
88 | def_cmd(commit, 1, 1), | 97 | def_cmd(commit, 1, 1), |
89 | def_cmd(diff, 1, 1), | 98 | def_cmd(diff, 1, 1), |
90 | def_cmd(log, 1, 1), | 99 | def_cmd(log, 1, 1), |
91 | def_cmd(patch, 1, 0), | 100 | def_cmd(patch, 1, 0), |
92 | def_cmd(refs, 1, 1), | 101 | def_cmd(refs, 1, 1), |
93 | def_cmd(repolist, 0, 0), | 102 | def_cmd(repolist, 0, 0), |
94 | def_cmd(snapshot, 1, 0), | 103 | def_cmd(snapshot, 1, 0), |
95 | def_cmd(summary, 1, 1), | 104 | def_cmd(summary, 1, 1), |
96 | def_cmd(tag, 1, 1), | 105 | def_cmd(tag, 1, 1), |
97 | def_cmd(tree, 1, 1), | 106 | def_cmd(tree, 1, 1), |
98 | }; | 107 | }; |
99 | int i; | 108 | int i; |
100 | 109 | ||
101 | if (ctx->qry.page == NULL) { | 110 | if (ctx->qry.page == NULL) { |
102 | if (ctx->repo) | 111 | if (ctx->repo) |
103 | ctx->qry.page = "summary"; | 112 | ctx->qry.page = "summary"; |
104 | else | 113 | else |
105 | ctx->qry.page = "repolist"; | 114 | ctx->qry.page = "repolist"; |
106 | } | 115 | } |
107 | 116 | ||
108 | for(i = 0; i < sizeof(cmds)/sizeof(*cmds); i++) | 117 | for(i = 0; i < sizeof(cmds)/sizeof(*cmds); i++) |
109 | if (!strcmp(ctx->qry.page, cmds[i].name)) | 118 | if (!strcmp(ctx->qry.page, cmds[i].name)) |
110 | return &cmds[i]; | 119 | return &cmds[i]; |
111 | return NULL; | 120 | return NULL; |
112 | } | 121 | } |
diff --git a/ui-shared.c b/ui-shared.c index 83758f7..d08ede9 100644 --- a/ui-shared.c +++ b/ui-shared.c | |||
@@ -85,64 +85,107 @@ const char *cgit_repobasename(const char *reponame) | |||
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) | ||
118 | { | ||
119 | char *delim = "?"; | ||
120 | |||
121 | if (ctx.cfg.virtual_root) { | ||
122 | html_attr(ctx.cfg.virtual_root); | ||
123 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') | ||
124 | html("/"); | ||
125 | } else | ||
126 | html(ctx.cfg.script_name); | ||
127 | |||
128 | if (page) { | ||
129 | htmlf("?p=%s", page); | ||
130 | delim = "&"; | ||
131 | } | ||
132 | if (search) { | ||
133 | html(delim); | ||
134 | html("q="); | ||
135 | html_attr(search); | ||
136 | } | ||
137 | } | ||
138 | |||
139 | static void site_link(char *page, char *name, char *title, char *class, | ||
140 | char *search) | ||
141 | { | ||
142 | html("<a"); | ||
143 | if (title) { | ||
144 | html(" title='"); | ||
145 | html_attr(title); | ||
146 | html("'"); | ||
147 | } | ||
148 | if (class) { | ||
149 | html(" class='"); | ||
150 | html_attr(class); | ||
151 | html("'"); | ||
152 | } | ||
153 | html(" href='"); | ||
154 | site_url(page, search); | ||
155 | html("'>"); | ||
156 | html_txt(name); | ||
157 | html("</a>"); | ||
158 | } | ||
159 | |||
117 | static char *repolink(char *title, char *class, char *page, char *head, | 160 | static char *repolink(char *title, char *class, char *page, char *head, |
118 | char *path) | 161 | char *path) |
119 | { | 162 | { |
120 | char *delim = "?"; | 163 | char *delim = "?"; |
121 | 164 | ||
122 | html("<a"); | 165 | html("<a"); |
123 | if (title) { | 166 | if (title) { |
124 | html(" title='"); | 167 | html(" title='"); |
125 | html_attr(title); | 168 | html_attr(title); |
126 | html("'"); | 169 | html("'"); |
127 | } | 170 | } |
128 | if (class) { | 171 | if (class) { |
129 | html(" class='"); | 172 | html(" class='"); |
130 | html_attr(class); | 173 | html_attr(class); |
131 | html("'"); | 174 | html("'"); |
132 | } | 175 | } |
133 | html(" href='"); | 176 | html(" href='"); |
134 | if (ctx.cfg.virtual_root) { | 177 | if (ctx.cfg.virtual_root) { |
135 | html_attr(ctx.cfg.virtual_root); | 178 | html_attr(ctx.cfg.virtual_root); |
136 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') | 179 | if (ctx.cfg.virtual_root[strlen(ctx.cfg.virtual_root) - 1] != '/') |
137 | html("/"); | 180 | html("/"); |
138 | html_attr(ctx.repo->url); | 181 | html_attr(ctx.repo->url); |
139 | if (ctx.repo->url[strlen(ctx.repo->url) - 1] != '/') | 182 | if (ctx.repo->url[strlen(ctx.repo->url) - 1] != '/') |
140 | html("/"); | 183 | html("/"); |
141 | if (page) { | 184 | if (page) { |
142 | html(page); | 185 | html(page); |
143 | html("/"); | 186 | html("/"); |
144 | if (path) | 187 | if (path) |
145 | html_attr(path); | 188 | html_attr(path); |
146 | } | 189 | } |
147 | } else { | 190 | } else { |
148 | html(ctx.cfg.script_name); | 191 | html(ctx.cfg.script_name); |
@@ -502,85 +545,89 @@ void cgit_print_pageheader(struct cgit_context *ctx) | |||
502 | html("</form>"); | 545 | html("</form>"); |
503 | } else | 546 | } else |
504 | html_txt(ctx->cfg.root_title); | 547 | html_txt(ctx->cfg.root_title); |
505 | html("</td></tr>\n"); | 548 | html("</td></tr>\n"); |
506 | 549 | ||
507 | html("<tr><td class='sub'"); | 550 | html("<tr><td class='sub'"); |
508 | if (ctx->repo) { | 551 | if (ctx->repo) { |
509 | html(" colspan='2'>"); | 552 | html(" colspan='2'>"); |
510 | html_txt(ctx->repo->desc); | 553 | html_txt(ctx->repo->desc); |
511 | } else { | 554 | } else { |
512 | html(">"); | 555 | html(">"); |
513 | if (ctx->cfg.root_desc) | 556 | if (ctx->cfg.root_desc) |
514 | html_txt(ctx->cfg.root_desc); | 557 | html_txt(ctx->cfg.root_desc); |
515 | else if (ctx->cfg.index_info) | 558 | else if (ctx->cfg.index_info) |
516 | html_include(ctx->cfg.index_info); | 559 | html_include(ctx->cfg.index_info); |
517 | } | 560 | } |
518 | html("</td></tr></table>\n"); | 561 | html("</td></tr></table>\n"); |
519 | 562 | ||
520 | html("<table class='tabs'><tr><td>\n"); | 563 | html("<table class='tabs'><tr><td>\n"); |
521 | if (ctx->repo) { | 564 | if (ctx->repo) { |
522 | reporevlink(NULL, "summary", NULL, hc(cmd, "summary"), | 565 | reporevlink(NULL, "summary", NULL, hc(cmd, "summary"), |
523 | ctx->qry.head, NULL, NULL); | 566 | ctx->qry.head, NULL, NULL); |
524 | cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, | 567 | cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, |
525 | ctx->qry.sha1, NULL); | 568 | ctx->qry.sha1, NULL); |
526 | cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, | 569 | cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, |
527 | NULL, NULL, 0, NULL, NULL); | 570 | NULL, NULL, 0, NULL, NULL); |
528 | cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, | 571 | cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, |
529 | ctx->qry.sha1, NULL); | 572 | ctx->qry.sha1, NULL); |
530 | cgit_commit_link("commit", NULL, hc(cmd, "commit"), | 573 | cgit_commit_link("commit", NULL, hc(cmd, "commit"), |
531 | ctx->qry.head, ctx->qry.sha1); | 574 | ctx->qry.head, ctx->qry.sha1); |
532 | cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, | 575 | cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, |
533 | ctx->qry.sha1, ctx->qry.sha2, NULL); | 576 | ctx->qry.sha1, ctx->qry.sha2, NULL); |
577 | if (ctx->repo->readme) | ||
578 | reporevlink("about", "about", NULL, | ||
579 | hc(cmd, "about"), ctx->qry.head, NULL, | ||
580 | NULL); | ||
534 | html("</td><td class='form'>"); | 581 | html("</td><td class='form'>"); |
535 | html("<form class='right' method='get' action='"); | 582 | html("<form class='right' method='get' action='"); |
536 | if (ctx->cfg.virtual_root) | 583 | if (ctx->cfg.virtual_root) |
537 | html_attr(cgit_fileurl(ctx->qry.repo, "log", | 584 | html_attr(cgit_fileurl(ctx->qry.repo, "log", |
538 | ctx->qry.path, NULL)); | 585 | ctx->qry.path, NULL)); |
539 | html("'>\n"); | 586 | html("'>\n"); |
540 | add_hidden_formfields(1, 0, "log"); | 587 | add_hidden_formfields(1, 0, "log"); |
541 | html("<select name='qt'>\n"); | 588 | html("<select name='qt'>\n"); |
542 | html_option("grep", "log msg", ctx->qry.grep); | 589 | html_option("grep", "log msg", ctx->qry.grep); |
543 | html_option("author", "author", ctx->qry.grep); | 590 | html_option("author", "author", ctx->qry.grep); |
544 | html_option("committer", "committer", ctx->qry.grep); | 591 | html_option("committer", "committer", ctx->qry.grep); |
545 | html("</select>\n"); | 592 | html("</select>\n"); |
546 | html("<input class='txt' type='text' size='10' name='q' value='"); | 593 | html("<input class='txt' type='text' size='10' name='q' value='"); |
547 | html_attr(ctx->qry.search); | 594 | html_attr(ctx->qry.search); |
548 | html("'/>\n"); | 595 | html("'/>\n"); |
549 | html("<input type='submit' value='search'/>\n"); | 596 | html("<input type='submit' value='search'/>\n"); |
550 | html("</form>\n"); | 597 | html("</form>\n"); |
551 | } else { | 598 | } else { |
552 | html("<a class='active' href='"); | 599 | site_link(NULL, "index", NULL, hc(cmd, "repolist"), NULL); |
553 | html_attr(cgit_rooturl()); | 600 | if (ctx->cfg.root_readme) |
554 | html("'>index</a>\n"); | 601 | site_link("about", "about", NULL, hc(cmd, "about"), NULL); |
555 | html("</td><td class='form'>"); | 602 | html("</td><td class='form'>"); |
556 | html("<form method='get' action='"); | 603 | html("<form method='get' action='"); |
557 | html_attr(cgit_rooturl()); | 604 | html_attr(cgit_rooturl()); |
558 | html("'>\n"); | 605 | html("'>\n"); |
559 | html("<input type='text' name='q' size='10' value='"); | 606 | html("<input type='text' name='q' size='10' value='"); |
560 | html_attr(ctx->qry.search); | 607 | html_attr(ctx->qry.search); |
561 | html("'/>\n"); | 608 | html("'/>\n"); |
562 | html("<input type='submit' value='search'/>\n"); | 609 | html("<input type='submit' value='search'/>\n"); |
563 | html("</form>"); | 610 | html("</form>"); |
564 | } | 611 | } |
565 | html("</td></tr></table>\n"); | 612 | html("</td></tr></table>\n"); |
566 | html("<div class='content'>"); | 613 | html("<div class='content'>"); |
567 | } | 614 | } |
568 | 615 | ||
569 | void cgit_print_filemode(unsigned short mode) | 616 | void cgit_print_filemode(unsigned short mode) |
570 | { | 617 | { |
571 | if (S_ISDIR(mode)) | 618 | if (S_ISDIR(mode)) |
572 | html("d"); | 619 | html("d"); |
573 | else if (S_ISLNK(mode)) | 620 | else if (S_ISLNK(mode)) |
574 | html("l"); | 621 | html("l"); |
575 | else if (S_ISGITLINK(mode)) | 622 | else if (S_ISGITLINK(mode)) |
576 | html("m"); | 623 | html("m"); |
577 | else | 624 | else |
578 | html("-"); | 625 | html("-"); |
579 | html_fileperm(mode >> 6); | 626 | html_fileperm(mode >> 6); |
580 | html_fileperm(mode >> 3); | 627 | html_fileperm(mode >> 3); |
581 | html_fileperm(mode); | 628 | html_fileperm(mode); |
582 | } | 629 | } |
583 | 630 | ||
584 | void cgit_print_snapshot_links(const char *repo, const char *head, | 631 | void cgit_print_snapshot_links(const char *repo, const char *head, |
585 | const char *hex, int snapshots) | 632 | const char *hex, int snapshots) |
586 | { | 633 | { |