summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2007-06-16 23:23:08 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2007-06-16 23:39:05 (UTC)
commit44947bfcdc0d6e8c7d673bea0538cbf2a182f289 (patch) (unidiff)
tree3f65d6842738a8314ed72d5575a0b523061556b7
parentf91b9696a36008c245a3195800ba0c5fa3e890f9 (diff)
downloadcgit-44947bfcdc0d6e8c7d673bea0538cbf2a182f289.zip
cgit-44947bfcdc0d6e8c7d673bea0538cbf2a182f289.tar.gz
cgit-44947bfcdc0d6e8c7d673bea0538cbf2a182f289.tar.bz2
Add and use cgit_tree_link()
This creates a new function used to generate links to 'tree' page and uses the function everywhere a link to the 'tree' page is generated. Signed-off-by: Lars Hjemli <hjemli@gmail.com>
Diffstat (more/less context) (show whitespace changes)
-rw-r--r--cgit.h3
-rw-r--r--ui-commit.c10
-rw-r--r--ui-repolist.c35
-rw-r--r--ui-shared.c64
-rw-r--r--ui-tree.c33
5 files changed, 107 insertions, 38 deletions
diff --git a/cgit.h b/cgit.h
index b93e2e8..9a19c97 100644
--- a/cgit.h
+++ b/cgit.h
@@ -108,119 +108,122 @@ extern char *cgit_root_title;
108extern char *cgit_css; 108extern char *cgit_css;
109extern char *cgit_logo; 109extern char *cgit_logo;
110extern char *cgit_index_header; 110extern char *cgit_index_header;
111extern char *cgit_logo_link; 111extern char *cgit_logo_link;
112extern char *cgit_module_link; 112extern char *cgit_module_link;
113extern char *cgit_agefile; 113extern char *cgit_agefile;
114extern char *cgit_virtual_root; 114extern char *cgit_virtual_root;
115extern char *cgit_script_name; 115extern char *cgit_script_name;
116extern char *cgit_cache_root; 116extern char *cgit_cache_root;
117extern char *cgit_repo_group; 117extern char *cgit_repo_group;
118 118
119extern int cgit_nocache; 119extern int cgit_nocache;
120extern int cgit_snapshots; 120extern int cgit_snapshots;
121extern int cgit_enable_log_filecount; 121extern int cgit_enable_log_filecount;
122extern int cgit_enable_log_linecount; 122extern int cgit_enable_log_linecount;
123extern int cgit_max_lock_attempts; 123extern int cgit_max_lock_attempts;
124extern int cgit_cache_root_ttl; 124extern int cgit_cache_root_ttl;
125extern int cgit_cache_repo_ttl; 125extern int cgit_cache_repo_ttl;
126extern int cgit_cache_dynamic_ttl; 126extern int cgit_cache_dynamic_ttl;
127extern int cgit_cache_static_ttl; 127extern int cgit_cache_static_ttl;
128extern int cgit_cache_max_create_time; 128extern int cgit_cache_max_create_time;
129extern int cgit_summary_log; 129extern int cgit_summary_log;
130 130
131extern int cgit_max_msg_len; 131extern int cgit_max_msg_len;
132extern int cgit_max_repodesc_len; 132extern int cgit_max_repodesc_len;
133extern int cgit_max_commit_count; 133extern int cgit_max_commit_count;
134 134
135extern int cgit_query_has_symref; 135extern int cgit_query_has_symref;
136extern int cgit_query_has_sha1; 136extern int cgit_query_has_sha1;
137 137
138extern char *cgit_querystring; 138extern char *cgit_querystring;
139extern char *cgit_query_repo; 139extern char *cgit_query_repo;
140extern char *cgit_query_page; 140extern char *cgit_query_page;
141extern char *cgit_query_search; 141extern char *cgit_query_search;
142extern char *cgit_query_head; 142extern char *cgit_query_head;
143extern char *cgit_query_sha1; 143extern char *cgit_query_sha1;
144extern char *cgit_query_sha2; 144extern char *cgit_query_sha2;
145extern char *cgit_query_path; 145extern char *cgit_query_path;
146extern char *cgit_query_name; 146extern char *cgit_query_name;
147extern int cgit_query_ofs; 147extern int cgit_query_ofs;
148 148
149extern int htmlfd; 149extern int htmlfd;
150 150
151extern int cgit_get_cmd_index(const char *cmd); 151extern int cgit_get_cmd_index(const char *cmd);
152extern struct repoinfo *cgit_get_repoinfo(const char *url); 152extern struct repoinfo *cgit_get_repoinfo(const char *url);
153extern void cgit_global_config_cb(const char *name, const char *value); 153extern void cgit_global_config_cb(const char *name, const char *value);
154extern void cgit_repo_config_cb(const char *name, const char *value); 154extern void cgit_repo_config_cb(const char *name, const char *value);
155extern void cgit_querystring_cb(const char *name, const char *value); 155extern void cgit_querystring_cb(const char *name, const char *value);
156 156
157extern int chk_zero(int result, char *msg); 157extern int chk_zero(int result, char *msg);
158extern int chk_positive(int result, char *msg); 158extern int chk_positive(int result, char *msg);
159 159
160extern int hextoint(char c); 160extern int hextoint(char c);
161 161
162extern void *cgit_free_commitinfo(struct commitinfo *info); 162extern void *cgit_free_commitinfo(struct commitinfo *info);
163 163
164extern int cgit_diff_files(const unsigned char *old_sha1, 164extern int cgit_diff_files(const unsigned char *old_sha1,
165 const unsigned char *new_sha1, 165 const unsigned char *new_sha1,
166 linediff_fn fn); 166 linediff_fn fn);
167 167
168extern void cgit_diff_tree(const unsigned char *old_sha1, 168extern void cgit_diff_tree(const unsigned char *old_sha1,
169 const unsigned char *new_sha1, 169 const unsigned char *new_sha1,
170 filepair_fn fn); 170 filepair_fn fn);
171 171
172extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); 172extern void cgit_diff_commit(struct commit *commit, filepair_fn fn);
173 173
174extern char *fmt(const char *format,...); 174extern char *fmt(const char *format,...);
175 175
176extern void html(const char *txt); 176extern void html(const char *txt);
177extern void htmlf(const char *format,...); 177extern void htmlf(const char *format,...);
178extern void html_txt(char *txt); 178extern void html_txt(char *txt);
179extern void html_ntxt(int len, char *txt); 179extern void html_ntxt(int len, char *txt);
180extern void html_attr(char *txt); 180extern void html_attr(char *txt);
181extern void html_hidden(char *name, char *value); 181extern void html_hidden(char *name, char *value);
182extern void html_link_open(char *url, char *title, char *class); 182extern void html_link_open(char *url, char *title, char *class);
183extern void html_link_close(void); 183extern void html_link_close(void);
184extern void html_filemode(unsigned short mode); 184extern void html_filemode(unsigned short mode);
185extern int html_include(const char *filename); 185extern int html_include(const char *filename);
186 186
187extern int cgit_read_config(const char *filename, configfn fn); 187extern int cgit_read_config(const char *filename, configfn fn);
188extern int cgit_parse_query(char *txt, configfn fn); 188extern int cgit_parse_query(char *txt, configfn fn);
189extern struct commitinfo *cgit_parse_commit(struct commit *commit); 189extern struct commitinfo *cgit_parse_commit(struct commit *commit);
190extern struct taginfo *cgit_parse_tag(struct tag *tag); 190extern struct taginfo *cgit_parse_tag(struct tag *tag);
191extern void cgit_parse_url(const char *url); 191extern void cgit_parse_url(const char *url);
192 192
193extern char *cache_safe_filename(const char *unsafe); 193extern char *cache_safe_filename(const char *unsafe);
194extern int cache_lock(struct cacheitem *item); 194extern int cache_lock(struct cacheitem *item);
195extern int cache_unlock(struct cacheitem *item); 195extern int cache_unlock(struct cacheitem *item);
196extern int cache_cancel_lock(struct cacheitem *item); 196extern int cache_cancel_lock(struct cacheitem *item);
197extern int cache_exist(struct cacheitem *item); 197extern int cache_exist(struct cacheitem *item);
198extern int cache_expired(struct cacheitem *item); 198extern int cache_expired(struct cacheitem *item);
199 199
200extern char *cgit_repourl(const char *reponame); 200extern char *cgit_repourl(const char *reponame);
201extern char *cgit_pageurl(const char *reponame, const char *pagename, 201extern char *cgit_pageurl(const char *reponame, const char *pagename,
202 const char *query); 202 const char *query);
203 203
204extern void cgit_tree_link(char *name, char *title, char *class, char *head,
205 char *rev, char *path);
206
204extern void cgit_print_error(char *msg); 207extern void cgit_print_error(char *msg);
205extern void cgit_print_date(time_t secs, char *format); 208extern void cgit_print_date(time_t secs, char *format);
206extern void cgit_print_age(time_t t, time_t max_relative, char *format); 209extern void cgit_print_age(time_t t, time_t max_relative, char *format);
207extern void cgit_print_docstart(char *title, struct cacheitem *item); 210extern void cgit_print_docstart(char *title, struct cacheitem *item);
208extern void cgit_print_docend(); 211extern void cgit_print_docend();
209extern void cgit_print_pageheader(char *title, int show_search); 212extern void cgit_print_pageheader(char *title, int show_search);
210extern void cgit_print_snapshot_start(const char *mimetype, 213extern void cgit_print_snapshot_start(const char *mimetype,
211 const char *filename, 214 const char *filename,
212 struct cacheitem *item); 215 struct cacheitem *item);
213 216
214extern void cgit_print_repolist(struct cacheitem *item); 217extern void cgit_print_repolist(struct cacheitem *item);
215extern void cgit_print_summary(); 218extern void cgit_print_summary();
216extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path, int pager); 219extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path, int pager);
217extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path); 220extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path);
218extern void cgit_print_tree(const char *rev, char *path); 221extern void cgit_print_tree(const char *rev, char *path);
219extern void cgit_print_commit(const char *hex); 222extern void cgit_print_commit(const char *hex);
220extern void cgit_print_diff(const char *head, const char *old_hex, const char *new_hex, 223extern void cgit_print_diff(const char *head, const char *old_hex, const char *new_hex,
221 char *path); 224 char *path);
222extern void cgit_print_snapshot(struct cacheitem *item, const char *hex, 225extern void cgit_print_snapshot(struct cacheitem *item, const char *hex,
223 const char *format, const char *prefix, 226 const char *format, const char *prefix,
224 const char *filename); 227 const char *filename);
225 228
226#endif /* CGIT_H */ 229#endif /* CGIT_H */
diff --git a/ui-commit.c b/ui-commit.c
index ed5384d..885f870 100644
--- a/ui-commit.c
+++ b/ui-commit.c
@@ -60,180 +60,182 @@ void print_fileinfo(struct fileinfo *info)
60 } 60 }
61 61
62 html("<tr>"); 62 html("<tr>");
63 htmlf("<td class='mode'>"); 63 htmlf("<td class='mode'>");
64 if (is_null_sha1(info->new_sha1)) { 64 if (is_null_sha1(info->new_sha1)) {
65 html_filemode(info->old_mode); 65 html_filemode(info->old_mode);
66 } else { 66 } else {
67 html_filemode(info->new_mode); 67 html_filemode(info->new_mode);
68 } 68 }
69 69
70 if (info->old_mode != info->new_mode && 70 if (info->old_mode != info->new_mode &&
71 !is_null_sha1(info->old_sha1) && 71 !is_null_sha1(info->old_sha1) &&
72 !is_null_sha1(info->new_sha1)) { 72 !is_null_sha1(info->new_sha1)) {
73 html("<span class='modechange'>["); 73 html("<span class='modechange'>[");
74 html_filemode(info->old_mode); 74 html_filemode(info->old_mode);
75 html("]</span>"); 75 html("]</span>");
76 } 76 }
77 htmlf("</td><td class='%s'>", class); 77 htmlf("</td><td class='%s'>", class);
78 query = fmt("id=%s&amp;id2=%s&amp;path=%s", sha1_to_hex(info->old_sha1), 78 query = fmt("id=%s&amp;id2=%s&amp;path=%s", sha1_to_hex(info->old_sha1),
79 sha1_to_hex(info->new_sha1), info->new_path); 79 sha1_to_hex(info->new_sha1), info->new_path);
80 html_link_open(cgit_pageurl(cgit_query_repo, "diff", query), 80 html_link_open(cgit_pageurl(cgit_query_repo, "diff", query),
81 NULL, NULL); 81 NULL, NULL);
82 if (info->status == DIFF_STATUS_COPIED || 82 if (info->status == DIFF_STATUS_COPIED ||
83 info->status == DIFF_STATUS_RENAMED) { 83 info->status == DIFF_STATUS_RENAMED) {
84 html_txt(info->new_path); 84 html_txt(info->new_path);
85 htmlf("</a> (%s from ", info->status == DIFF_STATUS_COPIED ? 85 htmlf("</a> (%s from ", info->status == DIFF_STATUS_COPIED ?
86 "copied" : "renamed"); 86 "copied" : "renamed");
87 query2 = fmt("id=%s", sha1_to_hex(info->old_sha1)); 87 query2 = fmt("id=%s", sha1_to_hex(info->old_sha1));
88 html_link_open(cgit_pageurl(cgit_query_repo, "view", query2), 88 html_link_open(cgit_pageurl(cgit_query_repo, "view", query2),
89 NULL, NULL); 89 NULL, NULL);
90 html_txt(info->old_path); 90 html_txt(info->old_path);
91 html("</a>)"); 91 html("</a>)");
92 } else { 92 } else {
93 html_txt(info->new_path); 93 html_txt(info->new_path);
94 html("</a>"); 94 html("</a>");
95 } 95 }
96 html("</td><td class='right'>"); 96 html("</td><td class='right'>");
97 htmlf("%d", info->added + info->removed); 97 htmlf("%d", info->added + info->removed);
98 html("</td><td class='graph'>"); 98 html("</td><td class='graph'>");
99 htmlf("<table width='%d%%'><tr>", (max_changes > 100 ? 100 : max_changes)); 99 htmlf("<table width='%d%%'><tr>", (max_changes > 100 ? 100 : max_changes));
100 htmlf("<td class='add' style='width: %.1f%%;'/>", 100 htmlf("<td class='add' style='width: %.1f%%;'/>",
101 info->added * 100.0 / max_changes); 101 info->added * 100.0 / max_changes);
102 htmlf("<td class='rem' style='width: %.1f%%;'/>", 102 htmlf("<td class='rem' style='width: %.1f%%;'/>",
103 info->removed * 100.0 / max_changes); 103 info->removed * 100.0 / max_changes);
104 htmlf("<td class='none' style='width: %.1f%%;'/>", 104 htmlf("<td class='none' style='width: %.1f%%;'/>",
105 (max_changes - info->removed - info->added) * 100.0 / max_changes); 105 (max_changes - info->removed - info->added) * 100.0 / max_changes);
106 html("</tr></table></td></tr>\n"); 106 html("</tr></table></td></tr>\n");
107} 107}
108 108
109void cgit_count_diff_lines(char *line, int len) 109void cgit_count_diff_lines(char *line, int len)
110{ 110{
111 if (line && (len > 0)) { 111 if (line && (len > 0)) {
112 if (line[0] == '+') 112 if (line[0] == '+')
113 lines_added++; 113 lines_added++;
114 else if (line[0] == '-') 114 else if (line[0] == '-')
115 lines_removed++; 115 lines_removed++;
116 } 116 }
117} 117}
118 118
119void inspect_filepair(struct diff_filepair *pair) 119void inspect_filepair(struct diff_filepair *pair)
120{ 120{
121 files++; 121 files++;
122 lines_added = 0; 122 lines_added = 0;
123 lines_removed = 0; 123 lines_removed = 0;
124 cgit_diff_files(pair->one->sha1, pair->two->sha1, cgit_count_diff_lines); 124 cgit_diff_files(pair->one->sha1, pair->two->sha1, cgit_count_diff_lines);
125 if (files >= slots) { 125 if (files >= slots) {
126 if (slots == 0) 126 if (slots == 0)
127 slots = 4; 127 slots = 4;
128 else 128 else
129 slots = slots * 2; 129 slots = slots * 2;
130 items = xrealloc(items, slots * sizeof(struct fileinfo)); 130 items = xrealloc(items, slots * sizeof(struct fileinfo));
131 } 131 }
132 items[files-1].status = pair->status; 132 items[files-1].status = pair->status;
133 hashcpy(items[files-1].old_sha1, pair->one->sha1); 133 hashcpy(items[files-1].old_sha1, pair->one->sha1);
134 hashcpy(items[files-1].new_sha1, pair->two->sha1); 134 hashcpy(items[files-1].new_sha1, pair->two->sha1);
135 items[files-1].old_mode = pair->one->mode; 135 items[files-1].old_mode = pair->one->mode;
136 items[files-1].new_mode = pair->two->mode; 136 items[files-1].new_mode = pair->two->mode;
137 items[files-1].old_path = xstrdup(pair->one->path); 137 items[files-1].old_path = xstrdup(pair->one->path);
138 items[files-1].new_path = xstrdup(pair->two->path); 138 items[files-1].new_path = xstrdup(pair->two->path);
139 items[files-1].added = lines_added; 139 items[files-1].added = lines_added;
140 items[files-1].removed = lines_removed; 140 items[files-1].removed = lines_removed;
141 if (lines_added + lines_removed > max_changes) 141 if (lines_added + lines_removed > max_changes)
142 max_changes = lines_added + lines_removed; 142 max_changes = lines_added + lines_removed;
143 total_adds += lines_added; 143 total_adds += lines_added;
144 total_rems += lines_removed; 144 total_rems += lines_removed;
145} 145}
146 146
147 147
148void cgit_print_commit(const char *hex) 148void cgit_print_commit(const char *hex)
149{ 149{
150 struct commit *commit, *parent; 150 struct commit *commit, *parent;
151 struct commitinfo *info; 151 struct commitinfo *info;
152 struct commit_list *p; 152 struct commit_list *p;
153 unsigned char sha1[20]; 153 unsigned char sha1[20];
154 char *query; 154 char *query;
155 char *filename; 155 char *filename;
156 char *tmp;
156 int i; 157 int i;
157 158
158 if (get_sha1(hex, sha1)) { 159 if (get_sha1(hex, sha1)) {
159 cgit_print_error(fmt("Bad object id: %s", hex)); 160 cgit_print_error(fmt("Bad object id: %s", hex));
160 return; 161 return;
161 } 162 }
162 commit = lookup_commit_reference(sha1); 163 commit = lookup_commit_reference(sha1);
163 if (!commit) { 164 if (!commit) {
164 cgit_print_error(fmt("Bad commit reference: %s", hex)); 165 cgit_print_error(fmt("Bad commit reference: %s", hex));
165 return; 166 return;
166 } 167 }
167 info = cgit_parse_commit(commit); 168 info = cgit_parse_commit(commit);
168 169
169 html("<table class='commit-info'>\n"); 170 html("<table class='commit-info'>\n");
170 html("<tr><th>author</th><td>"); 171 html("<tr><th>author</th><td>");
171 html_txt(info->author); 172 html_txt(info->author);
172 html(" "); 173 html(" ");
173 html_txt(info->author_email); 174 html_txt(info->author_email);
174 html("</td><td class='right'>"); 175 html("</td><td class='right'>");
175 cgit_print_date(info->author_date, FMT_LONGDATE); 176 cgit_print_date(info->author_date, FMT_LONGDATE);
176 html("</td></tr>\n"); 177 html("</td></tr>\n");
177 html("<tr><th>committer</th><td>"); 178 html("<tr><th>committer</th><td>");
178 html_txt(info->committer); 179 html_txt(info->committer);
179 html(" "); 180 html(" ");
180 html_txt(info->committer_email); 181 html_txt(info->committer_email);
181 html("</td><td class='right'>"); 182 html("</td><td class='right'>");
182 cgit_print_date(info->committer_date, FMT_LONGDATE); 183 cgit_print_date(info->committer_date, FMT_LONGDATE);
183 html("</td></tr>\n"); 184 html("</td></tr>\n");
184 html("<tr><th>tree</th><td colspan='2' class='sha1'><a href='"); 185 html("<tr><th>tree</th><td colspan='2' class='sha1'>");
185 query = fmt("h=%s", sha1_to_hex(commit->object.sha1)); 186 tmp = xstrdup(hex);
186 html_attr(cgit_pageurl(cgit_query_repo, "tree", query)); 187 cgit_tree_link(sha1_to_hex(commit->tree->object.sha1), NULL, NULL,
187 htmlf("'>%s</a></td></tr>\n", sha1_to_hex(commit->tree->object.sha1)); 188 cgit_query_head, tmp, NULL);
189 html("</td></tr>\n");
188 for (p = commit->parents; p ; p = p->next) { 190 for (p = commit->parents; p ; p = p->next) {
189 parent = lookup_commit_reference(p->item->object.sha1); 191 parent = lookup_commit_reference(p->item->object.sha1);
190 if (!parent) { 192 if (!parent) {
191 html("<tr><td colspan='3'>"); 193 html("<tr><td colspan='3'>");
192 cgit_print_error("Error reading parent commit"); 194 cgit_print_error("Error reading parent commit");
193 html("</td></tr>"); 195 html("</td></tr>");
194 continue; 196 continue;
195 } 197 }
196 html("<tr><th>parent</th>" 198 html("<tr><th>parent</th>"
197 "<td colspan='2' class='sha1'>" 199 "<td colspan='2' class='sha1'>"
198 "<a href='"); 200 "<a href='");
199 query = fmt("h=%s", sha1_to_hex(p->item->object.sha1)); 201 query = fmt("h=%s", sha1_to_hex(p->item->object.sha1));
200 html_attr(cgit_pageurl(cgit_query_repo, "commit", query)); 202 html_attr(cgit_pageurl(cgit_query_repo, "commit", query));
201 htmlf("'>%s</a> (<a href='", 203 htmlf("'>%s</a> (<a href='",
202 sha1_to_hex(p->item->object.sha1)); 204 sha1_to_hex(p->item->object.sha1));
203 query = fmt("id=%s&amp;id2=%s", sha1_to_hex(parent->tree->object.sha1), 205 query = fmt("id=%s&amp;id2=%s", sha1_to_hex(parent->tree->object.sha1),
204 sha1_to_hex(commit->tree->object.sha1)); 206 sha1_to_hex(commit->tree->object.sha1));
205 html_attr(cgit_pageurl(cgit_query_repo, "diff", query)); 207 html_attr(cgit_pageurl(cgit_query_repo, "diff", query));
206 html("'>diff</a>)</td></tr>"); 208 html("'>diff</a>)</td></tr>");
207 } 209 }
208 if (cgit_repo->snapshots) { 210 if (cgit_repo->snapshots) {
209 htmlf("<tr><th>download</th><td colspan='2' class='sha1'><a href='"); 211 htmlf("<tr><th>download</th><td colspan='2' class='sha1'><a href='");
210 filename = fmt("%s-%s.zip", cgit_query_repo, hex); 212 filename = fmt("%s-%s.zip", cgit_query_repo, hex);
211 html_attr(cgit_pageurl(cgit_query_repo, "snapshot", 213 html_attr(cgit_pageurl(cgit_query_repo, "snapshot",
212 fmt("id=%s&amp;name=%s", hex, filename))); 214 fmt("id=%s&amp;name=%s", hex, filename)));
213 htmlf("'>%s</a></td></tr>", filename); 215 htmlf("'>%s</a></td></tr>", filename);
214 } 216 }
215 html("</table>\n"); 217 html("</table>\n");
216 html("<div class='commit-subject'>"); 218 html("<div class='commit-subject'>");
217 html_txt(info->subject); 219 html_txt(info->subject);
218 html("</div>"); 220 html("</div>");
219 html("<div class='commit-msg'>"); 221 html("<div class='commit-msg'>");
220 html_txt(info->msg); 222 html_txt(info->msg);
221 html("</div>"); 223 html("</div>");
222 if (!(commit->parents && commit->parents->next && commit->parents->next->next)) { 224 if (!(commit->parents && commit->parents->next && commit->parents->next->next)) {
223 html("<div class='diffstat-header'>Diffstat</div>"); 225 html("<div class='diffstat-header'>Diffstat</div>");
224 html("<table class='diffstat'>"); 226 html("<table class='diffstat'>");
225 max_changes = 0; 227 max_changes = 0;
226 cgit_diff_commit(commit, inspect_filepair); 228 cgit_diff_commit(commit, inspect_filepair);
227 for(i = 0; i<files; i++) 229 for(i = 0; i<files; i++)
228 print_fileinfo(&items[i]); 230 print_fileinfo(&items[i]);
229 html("</table>"); 231 html("</table>");
230 html("<div class='diffstat-summary'>"); 232 html("<div class='diffstat-summary'>");
231 htmlf("%d files changed, %d insertions, %d deletions (", 233 htmlf("%d files changed, %d insertions, %d deletions (",
232 files, total_adds, total_rems); 234 files, total_adds, total_rems);
233 query = fmt("h=%s", hex); 235 query = fmt("h=%s", hex);
234 html_link_open(cgit_pageurl(cgit_query_repo, "diff", query), NULL, NULL); 236 html_link_open(cgit_pageurl(cgit_query_repo, "diff", query), NULL, NULL);
235 html("show diff</a>)"); 237 html("show diff</a>)");
236 html("</div>"); 238 html("</div>");
237 } 239 }
238 cgit_free_commitinfo(info); 240 cgit_free_commitinfo(info);
239} 241}
diff --git a/ui-repolist.c b/ui-repolist.c
index e5c6c20..8ade91a 100644
--- a/ui-repolist.c
+++ b/ui-repolist.c
@@ -1,103 +1,100 @@
1/* ui-repolist.c: functions for generating the repolist page 1/* ui-repolist.c: functions for generating the repolist page
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 <time.h> 10#include <time.h>
11 11
12 12
13time_t read_agefile(char *path) 13time_t read_agefile(char *path)
14{ 14{
15 FILE *f; 15 FILE *f;
16 static char buf[64], buf2[64]; 16 static char buf[64], buf2[64];
17 17
18 if (!(f = fopen(path, "r"))) 18 if (!(f = fopen(path, "r")))
19 return -1; 19 return -1;
20 fgets(buf, sizeof(buf), f); 20 fgets(buf, sizeof(buf), f);
21 fclose(f); 21 fclose(f);
22 if (parse_date(buf, buf2, sizeof(buf2))) 22 if (parse_date(buf, buf2, sizeof(buf2)))
23 return strtoul(buf2, NULL, 10); 23 return strtoul(buf2, NULL, 10);
24 else 24 else
25 return 0; 25 return 0;
26} 26}
27 27
28static void print_modtime(struct repoinfo *repo) 28static void print_modtime(struct repoinfo *repo)
29{ 29{
30 char *path; 30 char *path;
31 struct stat s; 31 struct stat s;
32 32
33 path = fmt("%s/%s", repo->path, cgit_agefile); 33 path = fmt("%s/%s", repo->path, cgit_agefile);
34 if (stat(path, &s) == 0) { 34 if (stat(path, &s) == 0) {
35 cgit_print_age(read_agefile(path), -1, NULL); 35 cgit_print_age(read_agefile(path), -1, NULL);
36 return; 36 return;
37 } 37 }
38 38
39 path = fmt("%s/refs/heads/%s", repo->path, repo->defbranch); 39 path = fmt("%s/refs/heads/%s", repo->path, repo->defbranch);
40 if (stat(path, &s) != 0) 40 if (stat(path, &s) != 0)
41 return; 41 return;
42 cgit_print_age(s.st_mtime, -1, NULL); 42 cgit_print_age(s.st_mtime, -1, NULL);
43} 43}
44 44
45void cgit_print_repolist(struct cacheitem *item) 45void cgit_print_repolist(struct cacheitem *item)
46{ 46{
47 struct repoinfo *repo;
48 int i; 47 int i;
49 char *last_group = NULL; 48 char *last_group = NULL;
50 49
51 cgit_print_docstart(cgit_root_title, item); 50 cgit_print_docstart(cgit_root_title, item);
52 cgit_print_pageheader(cgit_root_title, 0); 51 cgit_print_pageheader(cgit_root_title, 0);
53 52
54 html("<table class='list nowrap'>"); 53 html("<table class='list nowrap'>");
55 if (cgit_index_header) { 54 if (cgit_index_header) {
56 html("<tr class='nohover'><td colspan='5' class='include-block'>"); 55 html("<tr class='nohover'><td colspan='5' class='include-block'>");
57 html_include(cgit_index_header); 56 html_include(cgit_index_header);
58 html("</td></tr>"); 57 html("</td></tr>");
59 } 58 }
60 html("<tr class='nohover'>" 59 html("<tr class='nohover'>"
61 "<th class='left'>Name</th>" 60 "<th class='left'>Name</th>"
62 "<th class='left'>Description</th>" 61 "<th class='left'>Description</th>"
63 "<th class='left'>Owner</th>" 62 "<th class='left'>Owner</th>"
64 "<th class='left'>Idle</th>" 63 "<th class='left'>Idle</th>"
65 "<th>Links</th></tr>\n"); 64 "<th>Links</th></tr>\n");
66 65
67 for (i=0; i<cgit_repolist.count; i++) { 66 for (i=0; i<cgit_repolist.count; i++) {
68 repo = &cgit_repolist.repos[i]; 67 cgit_repo = &cgit_repolist.repos[i];
69 if ((last_group == NULL && repo->group != NULL) || 68 if ((last_group == NULL && cgit_repo->group != NULL) ||
70 (last_group != NULL && repo->group == NULL) || 69 (last_group != NULL && cgit_repo->group == NULL) ||
71 (last_group != NULL && repo->group!= NULL && 70 (last_group != NULL && cgit_repo->group != NULL &&
72 strcmp(repo->group, last_group))) { 71 strcmp(cgit_repo->group, last_group))) {
73 html("<tr class='nohover'><td colspan='4' class='repogroup'>"); 72 html("<tr class='nohover'><td colspan='4' class='repogroup'>");
74 html_txt(repo->group); 73 html_txt(cgit_repo->group);
75 html("</td></tr>"); 74 html("</td></tr>");
76 last_group = repo->group; 75 last_group = cgit_repo->group;
77 } 76 }
78 htmlf("<tr><td class='%s'>", 77 htmlf("<tr><td class='%s'>",
79 repo->group ? "sublevel-repo" : "toplevel-repo"); 78 cgit_repo->group ? "sublevel-repo" : "toplevel-repo");
80 html_link_open(cgit_repourl(repo->url), repo->desc, NULL); 79 html_link_open(cgit_repourl(cgit_repo->url), NULL, NULL);
81 html_txt(repo->name); 80 html_txt(cgit_repo->name);
82 html_link_close(); 81 html_link_close();
83 html("</td><td>"); 82 html("</td><td>");
84 html_ntxt(cgit_max_repodesc_len, repo->desc); 83 html_ntxt(cgit_max_repodesc_len, cgit_repo->desc);
85 html("</td><td>"); 84 html("</td><td>");
86 html_txt(repo->owner); 85 html_txt(cgit_repo->owner);
87 html("</td><td>"); 86 html("</td><td>");
88 print_modtime(repo); 87 print_modtime(cgit_repo);
89 html("</td><td>"); 88 html("</td><td>");
90 html_link_open(cgit_repourl(repo->url), 89 html_link_open(cgit_repourl(cgit_repo->url),
91 "Summary", "button"); 90 "Summary", "button");
92 html("S</a>"); 91 html("S</a>");
93 html_link_open(cgit_pageurl(repo->name, "log", NULL), 92 html_link_open(cgit_pageurl(cgit_repo->name, "log", NULL),
94 "Log", "button"); 93 "Log", "button");
95 html("L</a>"); 94 html("L</a>");
96 html_link_open(cgit_pageurl(repo->name, "tree", NULL), 95 cgit_tree_link("F", "Files", "button", NULL, NULL, NULL);
97 "Files", "button");
98 html("F</a>");
99 html("</td></tr>\n"); 96 html("</td></tr>\n");
100 } 97 }
101 html("</table>"); 98 html("</table>");
102 cgit_print_docend(); 99 cgit_print_docend();
103} 100}
diff --git a/ui-shared.c b/ui-shared.c
index aba93e8..9ab6409 100644
--- a/ui-shared.c
+++ b/ui-shared.c
@@ -1,185 +1,249 @@
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 10
11const char cgit_doctype[] = 11const char cgit_doctype[] =
12"<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\"\n" 12"<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\"\n"
13" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n"; 13" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n";
14 14
15static char *http_date(time_t t) 15static char *http_date(time_t t)
16{ 16{
17 static char day[][4] = 17 static char day[][4] =
18 {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; 18 {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
19 static char month[][4] = 19 static char month[][4] =
20 {"Jan", "Feb", "Mar", "Apr", "May", "Jun", 20 {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
21 "Jul", "Aug", "Sep", "Oct", "Now", "Dec"}; 21 "Jul", "Aug", "Sep", "Oct", "Now", "Dec"};
22 struct tm *tm = gmtime(&t); 22 struct tm *tm = gmtime(&t);
23 return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday], 23 return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday],
24 tm->tm_mday, month[tm->tm_mon], 1900+tm->tm_year, 24 tm->tm_mday, month[tm->tm_mon], 1900+tm->tm_year,
25 tm->tm_hour, tm->tm_min, tm->tm_sec); 25 tm->tm_hour, tm->tm_min, tm->tm_sec);
26} 26}
27 27
28static long ttl_seconds(long ttl) 28static long ttl_seconds(long ttl)
29{ 29{
30 if (ttl<0) 30 if (ttl<0)
31 return 60 * 60 * 24 * 365; 31 return 60 * 60 * 24 * 365;
32 else 32 else
33 return ttl * 60; 33 return ttl * 60;
34} 34}
35 35
36void cgit_print_error(char *msg) 36void cgit_print_error(char *msg)
37{ 37{
38 html("<div class='error'>"); 38 html("<div class='error'>");
39 html_txt(msg); 39 html_txt(msg);
40 html("</div>\n"); 40 html("</div>\n");
41} 41}
42 42
43char *cgit_rooturl() 43char *cgit_rooturl()
44{ 44{
45 if (cgit_virtual_root) 45 if (cgit_virtual_root)
46 return fmt("%s/", cgit_virtual_root); 46 return fmt("%s/", cgit_virtual_root);
47 else 47 else
48 return cgit_script_name; 48 return cgit_script_name;
49} 49}
50 50
51char *cgit_repourl(const char *reponame) 51char *cgit_repourl(const char *reponame)
52{ 52{
53 if (cgit_virtual_root) { 53 if (cgit_virtual_root) {
54 return fmt("%s/%s/", cgit_virtual_root, reponame); 54 return fmt("%s/%s/", cgit_virtual_root, reponame);
55 } else { 55 } else {
56 return fmt("?r=%s", reponame); 56 return fmt("?r=%s", reponame);
57 } 57 }
58} 58}
59 59
60char *cgit_pageurl(const char *reponame, const char *pagename, 60char *cgit_pageurl(const char *reponame, const char *pagename,
61 const char *query) 61 const char *query)
62{ 62{
63 if (cgit_virtual_root) { 63 if (cgit_virtual_root) {
64 if (query) 64 if (query)
65 return fmt("%s/%s/%s/?%s", cgit_virtual_root, reponame, 65 return fmt("%s/%s/%s/?%s", cgit_virtual_root, reponame,
66 pagename, query); 66 pagename, query);
67 else 67 else
68 return fmt("%s/%s/%s/", cgit_virtual_root, reponame, 68 return fmt("%s/%s/%s/", cgit_virtual_root, reponame,
69 pagename); 69 pagename);
70 } else { 70 } else {
71 if (query) 71 if (query)
72 return fmt("?r=%s&amp;p=%s&amp;%s", reponame, pagename, query); 72 return fmt("?r=%s&amp;p=%s&amp;%s", reponame, pagename, query);
73 else 73 else
74 return fmt("?r=%s&amp;p=%s", reponame, pagename); 74 return fmt("?r=%s&amp;p=%s", reponame, pagename);
75 } 75 }
76} 76}
77 77
78char *cgit_currurl() 78char *cgit_currurl()
79{ 79{
80 if (!cgit_virtual_root) 80 if (!cgit_virtual_root)
81 return cgit_script_name; 81 return cgit_script_name;
82 else if (cgit_query_page) 82 else if (cgit_query_page)
83 return fmt("%s/%s/%s/", cgit_virtual_root, cgit_query_repo, cgit_query_page); 83 return fmt("%s/%s/%s/", cgit_virtual_root, cgit_query_repo, cgit_query_page);
84 else if (cgit_query_repo) 84 else if (cgit_query_repo)
85 return fmt("%s/%s/", cgit_virtual_root, cgit_query_repo); 85 return fmt("%s/%s/", cgit_virtual_root, cgit_query_repo);
86 else 86 else
87 return fmt("%s/", cgit_virtual_root); 87 return fmt("%s/", cgit_virtual_root);
88} 88}
89 89
90static char *repolink(char *title, char *class, char *page, char *head,
91 char *path)
92{
93 char *delim = "?";
94
95 html("<a");
96 if (title) {
97 html(" title='");
98 html_attr(title);
99 html("'");
100 }
101 if (class) {
102 html(" class='");
103 html_attr(class);
104 html("'");
105 }
106 html(" href='");
107 if (cgit_virtual_root) {
108 html_attr(cgit_virtual_root);
109 if (cgit_virtual_root[strlen(cgit_virtual_root) - 1] != '/')
110 html("/");
111 html_attr(cgit_repo->url);
112 if (cgit_repo->url[strlen(cgit_repo->url) - 1] != '/')
113 html("/");
114 html(page);
115 html("/");
116 if (path)
117 html_attr(path);
118 } else {
119 html(cgit_script_name);
120 html("?url=");
121 html_attr(cgit_repo->url);
122 if (cgit_repo->url[strlen(cgit_repo->url) - 1] != '/')
123 html("/");
124 html(page);
125 html("/");
126 if (path)
127 html_attr(path);
128 delim = "&amp;";
129 }
130 if (head && head != cgit_query_head) {
131 html(delim);
132 html("h=");
133 html_attr(head);
134 delim = "&amp;";
135 }
136 return fmt("%s", delim);
137}
138
139void cgit_tree_link(char *name, char *title, char *class, char *head,
140 char *rev, char *path)
141{
142 char *delim;
143
144 delim = repolink(title, class, "tree", head, path);
145 if (rev && rev != cgit_query_head) {
146 html(delim);
147 html("id=");
148 html_attr(rev);
149 }
150 html("'>");
151 html_txt(name);
152 html("</a>");
153}
90 154
91void cgit_print_date(time_t secs, char *format) 155void cgit_print_date(time_t secs, char *format)
92{ 156{
93 char buf[64]; 157 char buf[64];
94 struct tm *time; 158 struct tm *time;
95 159
96 time = gmtime(&secs); 160 time = gmtime(&secs);
97 strftime(buf, sizeof(buf)-1, format, time); 161 strftime(buf, sizeof(buf)-1, format, time);
98 html_txt(buf); 162 html_txt(buf);
99} 163}
100 164
101void cgit_print_age(time_t t, time_t max_relative, char *format) 165void cgit_print_age(time_t t, time_t max_relative, char *format)
102{ 166{
103 time_t now, secs; 167 time_t now, secs;
104 168
105 time(&now); 169 time(&now);
106 secs = now - t; 170 secs = now - t;
107 171
108 if (secs > max_relative && max_relative >= 0) { 172 if (secs > max_relative && max_relative >= 0) {
109 cgit_print_date(t, format); 173 cgit_print_date(t, format);
110 return; 174 return;
111 } 175 }
112 176
113 if (secs < TM_HOUR * 2) { 177 if (secs < TM_HOUR * 2) {
114 htmlf("<span class='age-mins'>%.0f min.</span>", 178 htmlf("<span class='age-mins'>%.0f min.</span>",
115 secs * 1.0 / TM_MIN); 179 secs * 1.0 / TM_MIN);
116 return; 180 return;
117 } 181 }
118 if (secs < TM_DAY * 2) { 182 if (secs < TM_DAY * 2) {
119 htmlf("<span class='age-hours'>%.0f hours</span>", 183 htmlf("<span class='age-hours'>%.0f hours</span>",
120 secs * 1.0 / TM_HOUR); 184 secs * 1.0 / TM_HOUR);
121 return; 185 return;
122 } 186 }
123 if (secs < TM_WEEK * 2) { 187 if (secs < TM_WEEK * 2) {
124 htmlf("<span class='age-days'>%.0f days</span>", 188 htmlf("<span class='age-days'>%.0f days</span>",
125 secs * 1.0 / TM_DAY); 189 secs * 1.0 / TM_DAY);
126 return; 190 return;
127 } 191 }
128 if (secs < TM_MONTH * 2) { 192 if (secs < TM_MONTH * 2) {
129 htmlf("<span class='age-weeks'>%.0f weeks</span>", 193 htmlf("<span class='age-weeks'>%.0f weeks</span>",
130 secs * 1.0 / TM_WEEK); 194 secs * 1.0 / TM_WEEK);
131 return; 195 return;
132 } 196 }
133 if (secs < TM_YEAR * 2) { 197 if (secs < TM_YEAR * 2) {
134 htmlf("<span class='age-months'>%.0f months</span>", 198 htmlf("<span class='age-months'>%.0f months</span>",
135 secs * 1.0 / TM_MONTH); 199 secs * 1.0 / TM_MONTH);
136 return; 200 return;
137 } 201 }
138 htmlf("<span class='age-years'>%.0f years</span>", 202 htmlf("<span class='age-years'>%.0f years</span>",
139 secs * 1.0 / TM_YEAR); 203 secs * 1.0 / TM_YEAR);
140} 204}
141 205
142void cgit_print_docstart(char *title, struct cacheitem *item) 206void cgit_print_docstart(char *title, struct cacheitem *item)
143{ 207{
144 html("Content-Type: text/html; charset=utf-8\n"); 208 html("Content-Type: text/html; charset=utf-8\n");
145 htmlf("Last-Modified: %s\n", http_date(item->st.st_mtime)); 209 htmlf("Last-Modified: %s\n", http_date(item->st.st_mtime));
146 htmlf("Expires: %s\n", http_date(item->st.st_mtime + 210 htmlf("Expires: %s\n", http_date(item->st.st_mtime +
147 ttl_seconds(item->ttl))); 211 ttl_seconds(item->ttl)));
148 html("\n"); 212 html("\n");
149 html(cgit_doctype); 213 html(cgit_doctype);
150 html("<html>\n"); 214 html("<html>\n");
151 html("<head>\n"); 215 html("<head>\n");
152 html("<title>"); 216 html("<title>");
153 html_txt(title); 217 html_txt(title);
154 html("</title>\n"); 218 html("</title>\n");
155 htmlf("<meta name='generator' content='cgit v%s'/>\n", cgit_version); 219 htmlf("<meta name='generator' content='cgit v%s'/>\n", cgit_version);
156 html("<link rel='stylesheet' type='text/css' href='"); 220 html("<link rel='stylesheet' type='text/css' href='");
157 html_attr(cgit_css); 221 html_attr(cgit_css);
158 html("'/>\n"); 222 html("'/>\n");
159 html("</head>\n"); 223 html("</head>\n");
160 html("<body>\n"); 224 html("<body>\n");
161} 225}
162 226
163void cgit_print_docend() 227void cgit_print_docend()
164{ 228{
165 html("</td></tr></table>"); 229 html("</td></tr></table>");
166 html("</body>\n</html>\n"); 230 html("</body>\n</html>\n");
167} 231}
168 232
169void cgit_print_pageheader(char *title, int show_search) 233void cgit_print_pageheader(char *title, int show_search)
170{ 234{
171 html("<table id='layout'>"); 235 html("<table id='layout'>");
172 html("<tr><td id='header'>"); 236 html("<tr><td id='header'>");
173 html(cgit_root_title); 237 html(cgit_root_title);
174 html("</td><td id='logo'>"); 238 html("</td><td id='logo'>");
175 html("<a href='"); 239 html("<a href='");
176 html_attr(cgit_logo_link); 240 html_attr(cgit_logo_link);
177 htmlf("'><img src='%s' alt='logo'/></a>", cgit_logo); 241 htmlf("'><img src='%s' alt='logo'/></a>", cgit_logo);
178 html("</td></tr>"); 242 html("</td></tr>");
179 html("<tr><td id='crumb'>"); 243 html("<tr><td id='crumb'>");
180 htmlf("<a href='%s'>root</a>", cgit_rooturl()); 244 htmlf("<a href='%s'>root</a>", cgit_rooturl());
181 if (cgit_query_repo) { 245 if (cgit_query_repo) {
182 htmlf(" : <a href='%s'>", cgit_repourl(cgit_repo->url)); 246 htmlf(" : <a href='%s'>", cgit_repourl(cgit_repo->url));
183 html_txt(cgit_repo->name); 247 html_txt(cgit_repo->name);
184 htmlf("</a> : %s", title); 248 htmlf("</a> : %s", title);
185 } 249 }
diff --git a/ui-tree.c b/ui-tree.c
index db0bef6..3b82374 100644
--- a/ui-tree.c
+++ b/ui-tree.c
@@ -1,213 +1,216 @@
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 10
11char *curr_rev; 11char *curr_rev;
12char *match_path; 12char *match_path;
13int header = 0; 13int header = 0;
14 14
15static void print_object(const unsigned char *sha1, char *path) 15static void print_object(const unsigned char *sha1, char *path)
16{ 16{
17 enum object_type type; 17 enum object_type type;
18 unsigned char *buf; 18 unsigned char *buf;
19 unsigned long size, lineno, start, idx; 19 unsigned long size, lineno, start, idx;
20 20
21 type = sha1_object_info(sha1, &size); 21 type = sha1_object_info(sha1, &size);
22 if (type == OBJ_BAD) { 22 if (type == OBJ_BAD) {
23 cgit_print_error(fmt("Bad object name: %s", 23 cgit_print_error(fmt("Bad object name: %s",
24 sha1_to_hex(sha1))); 24 sha1_to_hex(sha1)));
25 return; 25 return;
26 } 26 }
27 27
28 buf = read_sha1_file(sha1, &type, &size); 28 buf = read_sha1_file(sha1, &type, &size);
29 if (!buf) { 29 if (!buf) {
30 cgit_print_error(fmt("Error reading object %s", 30 cgit_print_error(fmt("Error reading object %s",
31 sha1_to_hex(sha1))); 31 sha1_to_hex(sha1)));
32 return; 32 return;
33 } 33 }
34 34
35 html("<table class='blob'>\n"); 35 html("<table class='blob'>\n");
36 idx = 0; 36 idx = 0;
37 start = 0; 37 start = 0;
38 lineno = 0; 38 lineno = 0;
39 while(idx < size) { 39 while(idx < size) {
40 if (buf[idx] == '\n') { 40 if (buf[idx] == '\n') {
41 buf[idx] = '\0'; 41 buf[idx] = '\0';
42 htmlf("<tr><td class='no'>%d</td><td class='txt'>", 42 htmlf("<tr><td class='no'>%d</td><td class='txt'>",
43 ++lineno); 43 ++lineno);
44 html_txt(buf + start); 44 html_txt(buf + start);
45 html("</td></tr>\n"); 45 html("</td></tr>\n");
46 start = idx + 1; 46 start = idx + 1;
47 } 47 }
48 idx++; 48 idx++;
49 } 49 }
50 html("</table>\n"); 50 html("</table>\n");
51} 51}
52 52
53 53
54static int ls_item(const unsigned char *sha1, const char *base, int baselen, 54static int ls_item(const unsigned char *sha1, const char *base, int baselen,
55 const char *pathname, unsigned int mode, int stage) 55 const char *pathname, unsigned int mode, int stage)
56{ 56{
57 char *name; 57 char *name;
58 char *fullpath;
58 enum object_type type; 59 enum object_type type;
59 unsigned long size = 0; 60 unsigned long size = 0;
60 char *url, *qry; 61 char *url, *qry;
61 62
62 name = xstrdup(pathname); 63 name = xstrdup(pathname);
64 fullpath = fmt("%s%s%s", cgit_query_path ? cgit_query_path : "",
65 cgit_query_path ? "/" : "", name);
66
63 type = sha1_object_info(sha1, &size); 67 type = sha1_object_info(sha1, &size);
64 if (type == OBJ_BAD && !S_ISDIRLNK(mode)) { 68 if (type == OBJ_BAD && !S_ISDIRLNK(mode)) {
65 htmlf("<tr><td colspan='3'>Bad object: %s %s</td></tr>", 69 htmlf("<tr><td colspan='3'>Bad object: %s %s</td></tr>",
66 name, 70 name,
67 sha1_to_hex(sha1)); 71 sha1_to_hex(sha1));
68 return 0; 72 return 0;
69 } 73 }
70 qry = fmt("h=%s&amp;path=%s%s%s", curr_rev, 74
71 cgit_query_path ? cgit_query_path : "",
72 cgit_query_path ? "/" : "", pathname);
73 url = cgit_pageurl(cgit_query_repo, "tree", qry);
74 html("<tr><td class='filemode'>"); 75 html("<tr><td class='filemode'>");
75 html_filemode(mode); 76 html_filemode(mode);
76 html("</td><td "); 77 html("</td><td ");
77 if (S_ISDIRLNK(mode)) { 78 if (S_ISDIRLNK(mode)) {
78 htmlf("class='ls-mod'><a href='"); 79 htmlf("class='ls-mod'><a href='");
79 html_attr(fmt(cgit_repo->module_link, 80 html_attr(fmt(cgit_repo->module_link,
80 name, 81 name,
81 sha1_to_hex(sha1))); 82 sha1_to_hex(sha1)));
83 html("'>");
84 html_txt(name);
85 html("</a>");
82 } else if (S_ISDIR(mode)) { 86 } else if (S_ISDIR(mode)) {
83 html("class='ls-dir'><a href='"); 87 html("class='ls-dir'>");
84 html_attr(url); 88 cgit_tree_link(name, NULL, NULL, cgit_query_head,
89 curr_rev, fullpath);
85 } else { 90 } else {
86 html("class='ls-blob'><a href='"); 91 html("class='ls-blob'>");
87 html_attr(url); 92 cgit_tree_link(name, NULL, NULL, cgit_query_head,
93 curr_rev, fullpath);
88 } 94 }
89 htmlf("'>%s</a></td>", name); 95 htmlf("</td><td class='filesize'>%li</td>", size);
90 htmlf("<td class='filesize'>%li</td>", size);
91 96
92 html("<td class='links'><a href='"); 97 html("<td class='links'><a href='");
93 qry = fmt("h=%s&amp;path=%s%s%s", curr_rev, 98 qry = fmt("h=%s&amp;path=%s%s%s", curr_rev,
94 cgit_query_path ? cgit_query_path : "", 99 cgit_query_path ? cgit_query_path : "",
95 cgit_query_path ? "/" : "", pathname); 100 cgit_query_path ? "/" : "", pathname);
96 url = cgit_pageurl(cgit_query_repo, "log", qry); 101 url = cgit_pageurl(cgit_query_repo, "log", qry);
97 html_attr(url); 102 html_attr(url);
98 html("' class='button'>H</a></td>"); 103 html("' class='button'>H</a></td>");
99 html("</tr>\n"); 104 html("</tr>\n");
100 free(name); 105 free(name);
101 return 0; 106 return 0;
102} 107}
103 108
104static void ls_head() 109static void ls_head()
105{ 110{
106 html("<table class='list'>\n"); 111 html("<table class='list'>\n");
107 html("<tr class='nohover'>"); 112 html("<tr class='nohover'>");
108 html("<th class='left'>Mode</th>"); 113 html("<th class='left'>Mode</th>");
109 html("<th class='left'>Name</th>"); 114 html("<th class='left'>Name</th>");
110 html("<th class='right'>Size</th>"); 115 html("<th class='right'>Size</th>");
111 html("<th/>"); 116 html("<th/>");
112 html("</tr>\n"); 117 html("</tr>\n");
113 header = 1; 118 header = 1;
114} 119}
115 120
116static void ls_tail() 121static void ls_tail()
117{ 122{
118 if (!header) 123 if (!header)
119 return; 124 return;
120 html("</table>\n"); 125 html("</table>\n");
121 header = 0; 126 header = 0;
122} 127}
123 128
124static void ls_tree(const unsigned char *sha1, char *path) 129static void ls_tree(const unsigned char *sha1, char *path)
125{ 130{
126 struct tree *tree; 131 struct tree *tree;
127 132
128 tree = parse_tree_indirect(sha1); 133 tree = parse_tree_indirect(sha1);
129 if (!tree) { 134 if (!tree) {
130 cgit_print_error(fmt("Not a tree object: %s", 135 cgit_print_error(fmt("Not a tree object: %s",
131 sha1_to_hex(sha1))); 136 sha1_to_hex(sha1)));
132 return; 137 return;
133 } 138 }
134 139
135 ls_head(); 140 ls_head();
136 read_tree_recursive(tree, "", 0, 1, NULL, ls_item); 141 read_tree_recursive(tree, "", 0, 1, NULL, ls_item);
137 ls_tail(); 142 ls_tail();
138} 143}
139 144
140 145
141static int walk_tree(const unsigned char *sha1, const char *base, int baselen, 146static int walk_tree(const unsigned char *sha1, const char *base, int baselen,
142 const char *pathname, unsigned mode, int stage) 147 const char *pathname, unsigned mode, int stage)
143{ 148{
144 static int state; 149 static int state;
145 static char buffer[PATH_MAX]; 150 static char buffer[PATH_MAX];
146 char *url; 151 char *url;
147 152
148 if (state == 0) { 153 if (state == 0) {
149 memcpy(buffer, base, baselen); 154 memcpy(buffer, base, baselen);
150 strcpy(buffer+baselen, pathname); 155 strcpy(buffer+baselen, pathname);
151 url = cgit_pageurl(cgit_query_repo, "tree", 156 url = cgit_pageurl(cgit_query_repo, "tree",
152 fmt("h=%s&amp;path=%s", curr_rev, buffer)); 157 fmt("h=%s&amp;path=%s", curr_rev, buffer));
153 htmlf(" / <a href='"); 158 html("/");
154 html_attr(url); 159 cgit_tree_link(xstrdup(pathname), NULL, NULL, cgit_query_head,
155 html("'>"); 160 curr_rev, buffer);
156 html_txt(xstrdup(pathname));
157 html("</a>");
158 161
159 if (strcmp(match_path, buffer)) 162 if (strcmp(match_path, buffer))
160 return READ_TREE_RECURSIVE; 163 return READ_TREE_RECURSIVE;
161 164
162 if (S_ISDIR(mode)) { 165 if (S_ISDIR(mode)) {
163 state = 1; 166 state = 1;
164 ls_head(); 167 ls_head();
165 return READ_TREE_RECURSIVE; 168 return READ_TREE_RECURSIVE;
166 } else { 169 } else {
167 print_object(sha1, buffer); 170 print_object(sha1, buffer);
168 return 0; 171 return 0;
169 } 172 }
170 } 173 }
171 ls_item(sha1, base, baselen, pathname, mode, stage); 174 ls_item(sha1, base, baselen, pathname, mode, stage);
172 return 0; 175 return 0;
173} 176}
174 177
175 178
176/* 179/*
177 * Show a tree or a blob 180 * Show a tree or a blob
178 * rev: the commit pointing at the root tree object 181 * rev: the commit pointing at the root tree object
179 * path: path to tree or blob 182 * path: path to tree or blob
180 */ 183 */
181void cgit_print_tree(const char *rev, char *path) 184void cgit_print_tree(const char *rev, char *path)
182{ 185{
183 unsigned char sha1[20]; 186 unsigned char sha1[20];
184 struct commit *commit; 187 struct commit *commit;
185 const char *paths[] = {path, NULL}; 188 const char *paths[] = {path, NULL};
186 189
187 if (!rev) 190 if (!rev)
188 rev = cgit_query_head; 191 rev = cgit_query_head;
189 192
190 curr_rev = xstrdup(rev); 193 curr_rev = xstrdup(rev);
191 if (get_sha1(rev, sha1)) { 194 if (get_sha1(rev, sha1)) {
192 cgit_print_error(fmt("Invalid revision name: %s", rev)); 195 cgit_print_error(fmt("Invalid revision name: %s", rev));
193 return; 196 return;
194 } 197 }
195 commit = lookup_commit_reference(sha1); 198 commit = lookup_commit_reference(sha1);
196 if (!commit || parse_commit(commit)) { 199 if (!commit || parse_commit(commit)) {
197 cgit_print_error(fmt("Invalid commit reference: %s", rev)); 200 cgit_print_error(fmt("Invalid commit reference: %s", rev));
198 return; 201 return;
199 } 202 }
200 203
201 html("path: <a href='"); 204 html("path: <a href='");
202 html_attr(cgit_pageurl(cgit_query_repo, "tree", fmt("h=%s", rev))); 205 html_attr(cgit_pageurl(cgit_query_repo, "tree", fmt("h=%s", rev)));
203 html("'>root</a>"); 206 html("'>root</a>");
204 207
205 if (path == NULL) { 208 if (path == NULL) {
206 ls_tree(commit->tree->object.sha1, NULL); 209 ls_tree(commit->tree->object.sha1, NULL);
207 return; 210 return;
208 } 211 }
209 212
210 match_path = path; 213 match_path = path;
211 read_tree_recursive(commit->tree, NULL, 0, 0, paths, walk_tree); 214 read_tree_recursive(commit->tree, NULL, 0, 0, paths, walk_tree);
212 ls_tail(); 215 ls_tail();
213} 216}