author | Lars Hjemli <hjemli@gmail.com> | 2008-04-13 09:57:10 (UTC) |
---|---|---|
committer | Lars Hjemli <hjemli@gmail.com> | 2008-04-13 10:10:03 (UTC) |
commit | 536b0541fcfea2169e4df33043cd9ff14c657bce (patch) (unidiff) | |
tree | c27848059cd256541d79b01ab91fa07897a5da7d | |
parent | 931fc6d1e4986a4566647dda16af09bf69a28b89 (diff) | |
download | cgit-536b0541fcfea2169e4df33043cd9ff14c657bce.zip cgit-536b0541fcfea2169e4df33043cd9ff14c657bce.tar.gz cgit-536b0541fcfea2169e4df33043cd9ff14c657bce.tar.bz2 |
Implement minimal freetext search in the repolist
This makes the repolist much more usable when there's a lot of repositories
registered in cgitrc.
Signed-off-by: Lars Hjemli <hjemli@gmail.com>
-rw-r--r-- | ui-repolist.c | 49 | ||||
-rw-r--r-- | ui-shared.c | 11 |
2 files changed, 47 insertions, 13 deletions
diff --git a/ui-repolist.c b/ui-repolist.c index 9eba222..a7de453 100644 --- a/ui-repolist.c +++ b/ui-repolist.c | |||
@@ -15,100 +15,125 @@ | |||
15 | time_t read_agefile(char *path) | 15 | time_t read_agefile(char *path) |
16 | { | 16 | { |
17 | FILE *f; | 17 | FILE *f; |
18 | static char buf[64], buf2[64]; | 18 | static char buf[64], buf2[64]; |
19 | 19 | ||
20 | if (!(f = fopen(path, "r"))) | 20 | if (!(f = fopen(path, "r"))) |
21 | return -1; | 21 | return -1; |
22 | fgets(buf, sizeof(buf), f); | 22 | fgets(buf, sizeof(buf), f); |
23 | fclose(f); | 23 | fclose(f); |
24 | if (parse_date(buf, buf2, sizeof(buf2))) | 24 | if (parse_date(buf, buf2, sizeof(buf2))) |
25 | return strtoul(buf2, NULL, 10); | 25 | return strtoul(buf2, NULL, 10); |
26 | else | 26 | else |
27 | return 0; | 27 | return 0; |
28 | } | 28 | } |
29 | 29 | ||
30 | static void print_modtime(struct cgit_repo *repo) | 30 | static void print_modtime(struct cgit_repo *repo) |
31 | { | 31 | { |
32 | char *path; | 32 | char *path; |
33 | struct stat s; | 33 | struct stat s; |
34 | 34 | ||
35 | path = fmt("%s/%s", repo->path, ctx.cfg.agefile); | 35 | path = fmt("%s/%s", repo->path, ctx.cfg.agefile); |
36 | if (stat(path, &s) == 0) { | 36 | if (stat(path, &s) == 0) { |
37 | cgit_print_age(read_agefile(path), -1, NULL); | 37 | cgit_print_age(read_agefile(path), -1, NULL); |
38 | return; | 38 | return; |
39 | } | 39 | } |
40 | 40 | ||
41 | path = fmt("%s/refs/heads/%s", repo->path, repo->defbranch); | 41 | path = fmt("%s/refs/heads/%s", repo->path, repo->defbranch); |
42 | if (stat(path, &s) != 0) | 42 | if (stat(path, &s) != 0) |
43 | return; | 43 | return; |
44 | cgit_print_age(s.st_mtime, -1, NULL); | 44 | cgit_print_age(s.st_mtime, -1, NULL); |
45 | } | 45 | } |
46 | 46 | ||
47 | void cgit_print_repolist() | 47 | int is_match(struct cgit_repo *repo) |
48 | { | 48 | { |
49 | int i, columns = 4; | 49 | if (!ctx.qry.search) |
50 | char *last_group = NULL; | 50 | return 1; |
51 | 51 | if (repo->url && strstr(repo->url, ctx.qry.search)) | |
52 | if (ctx.cfg.enable_index_links) | 52 | return 1; |
53 | columns++; | 53 | if (repo->name && strstr(repo->name, ctx.qry.search)) |
54 | 54 | return 1; | |
55 | ctx.page.title = ctx.cfg.root_title; | 55 | if (repo->desc && strstr(repo->desc, ctx.qry.search)) |
56 | cgit_print_http_headers(&ctx); | 56 | return 1; |
57 | cgit_print_docstart(&ctx); | 57 | if (repo->owner && strstr(repo->owner, ctx.qry.search)) |
58 | cgit_print_pageheader(&ctx); | 58 | return 1; |
59 | return 0; | ||
60 | } | ||
59 | 61 | ||
60 | html("<table summary='repository list' class='list nowrap'>"); | 62 | void print_header(int columns) |
63 | { | ||
61 | if (ctx.cfg.index_header) { | 64 | if (ctx.cfg.index_header) { |
62 | htmlf("<tr class='nohover'><td colspan='%d' class='include-block'>", | 65 | htmlf("<tr class='nohover'><td colspan='%d' class='include-block'>", |
63 | columns); | 66 | columns); |
64 | html_include(ctx.cfg.index_header); | 67 | html_include(ctx.cfg.index_header); |
65 | html("</td></tr>"); | 68 | html("</td></tr>"); |
66 | } | 69 | } |
67 | html("<tr class='nohover'>" | 70 | html("<tr class='nohover'>" |
68 | "<th class='left'>Name</th>" | 71 | "<th class='left'>Name</th>" |
69 | "<th class='left'>Description</th>" | 72 | "<th class='left'>Description</th>" |
70 | "<th class='left'>Owner</th>" | 73 | "<th class='left'>Owner</th>" |
71 | "<th class='left'>Idle</th>"); | 74 | "<th class='left'>Idle</th>"); |
72 | if (ctx.cfg.enable_index_links) | 75 | if (ctx.cfg.enable_index_links) |
73 | html("<th class='left'>Links</th>"); | 76 | html("<th class='left'>Links</th>"); |
74 | html("</tr>\n"); | 77 | html("</tr>\n"); |
78 | } | ||
75 | 79 | ||
80 | void cgit_print_repolist() | ||
81 | { | ||
82 | int i, columns = 4, hits = 0, header = 0; | ||
83 | char *last_group = NULL; | ||
84 | |||
85 | if (ctx.cfg.enable_index_links) | ||
86 | columns++; | ||
87 | |||
88 | ctx.page.title = ctx.cfg.root_title; | ||
89 | cgit_print_http_headers(&ctx); | ||
90 | cgit_print_docstart(&ctx); | ||
91 | cgit_print_pageheader(&ctx); | ||
92 | |||
93 | html("<table summary='repository list' class='list nowrap'>"); | ||
76 | for (i=0; i<cgit_repolist.count; i++) { | 94 | for (i=0; i<cgit_repolist.count; i++) { |
77 | ctx.repo = &cgit_repolist.repos[i]; | 95 | ctx.repo = &cgit_repolist.repos[i]; |
96 | if (!is_match(ctx.repo)) | ||
97 | continue; | ||
98 | if (!header++) | ||
99 | print_header(columns); | ||
100 | hits++; | ||
78 | if ((last_group == NULL && ctx.repo->group != NULL) || | 101 | if ((last_group == NULL && ctx.repo->group != NULL) || |
79 | (last_group != NULL && ctx.repo->group == NULL) || | 102 | (last_group != NULL && ctx.repo->group == NULL) || |
80 | (last_group != NULL && ctx.repo->group != NULL && | 103 | (last_group != NULL && ctx.repo->group != NULL && |
81 | strcmp(ctx.repo->group, last_group))) { | 104 | strcmp(ctx.repo->group, last_group))) { |
82 | htmlf("<tr class='nohover'><td colspan='%d' class='repogroup'>", | 105 | htmlf("<tr class='nohover'><td colspan='%d' class='repogroup'>", |
83 | columns); | 106 | columns); |
84 | html_txt(ctx.repo->group); | 107 | html_txt(ctx.repo->group); |
85 | html("</td></tr>"); | 108 | html("</td></tr>"); |
86 | last_group = ctx.repo->group; | 109 | last_group = ctx.repo->group; |
87 | } | 110 | } |
88 | htmlf("<tr><td class='%s'>", | 111 | htmlf("<tr><td class='%s'>", |
89 | ctx.repo->group ? "sublevel-repo" : "toplevel-repo"); | 112 | ctx.repo->group ? "sublevel-repo" : "toplevel-repo"); |
90 | html_link_open(cgit_repourl(ctx.repo->url), NULL, NULL); | 113 | html_link_open(cgit_repourl(ctx.repo->url), NULL, NULL); |
91 | html_txt(ctx.repo->name); | 114 | html_txt(ctx.repo->name); |
92 | html_link_close(); | 115 | html_link_close(); |
93 | html("</td><td>"); | 116 | html("</td><td>"); |
94 | html_ntxt(ctx.cfg.max_repodesc_len, ctx.repo->desc); | 117 | html_ntxt(ctx.cfg.max_repodesc_len, ctx.repo->desc); |
95 | html("</td><td>"); | 118 | html("</td><td>"); |
96 | html_txt(ctx.repo->owner); | 119 | html_txt(ctx.repo->owner); |
97 | html("</td><td>"); | 120 | html("</td><td>"); |
98 | print_modtime(ctx.repo); | 121 | print_modtime(ctx.repo); |
99 | html("</td>"); | 122 | html("</td>"); |
100 | if (ctx.cfg.enable_index_links) { | 123 | if (ctx.cfg.enable_index_links) { |
101 | html("<td>"); | 124 | html("<td>"); |
102 | html_link_open(cgit_repourl(ctx.repo->url), | 125 | html_link_open(cgit_repourl(ctx.repo->url), |
103 | NULL, "button"); | 126 | NULL, "button"); |
104 | html("summary</a>"); | 127 | html("summary</a>"); |
105 | cgit_log_link("log", NULL, "button", NULL, NULL, NULL, | 128 | cgit_log_link("log", NULL, "button", NULL, NULL, NULL, |
106 | 0, NULL, NULL); | 129 | 0, NULL, NULL); |
107 | cgit_tree_link("tree", NULL, "button", NULL, NULL, NULL); | 130 | cgit_tree_link("tree", NULL, "button", NULL, NULL, NULL); |
108 | html("</td>"); | 131 | html("</td>"); |
109 | } | 132 | } |
110 | html("</tr>\n"); | 133 | html("</tr>\n"); |
111 | } | 134 | } |
112 | html("</table>"); | 135 | html("</table>"); |
136 | if (!hits) | ||
137 | cgit_print_error("No repositories found"); | ||
113 | cgit_print_docend(); | 138 | cgit_print_docend(); |
114 | } | 139 | } |
diff --git a/ui-shared.c b/ui-shared.c index 782caa7..6253a90 100644 --- a/ui-shared.c +++ b/ui-shared.c | |||
@@ -526,73 +526,82 @@ void cgit_print_pageheader(struct cgit_context *ctx) | |||
526 | 526 | ||
527 | html("</tr>\n"); | 527 | html("</tr>\n"); |
528 | html("</table>\n"); | 528 | html("</table>\n"); |
529 | 529 | ||
530 | html("<table class='tabs'><tr><td>\n"); | 530 | html("<table class='tabs'><tr><td>\n"); |
531 | if (ctx->repo) { | 531 | if (ctx->repo) { |
532 | reporevlink(NULL, "summary", NULL, hc(cmd, "summary"), | 532 | reporevlink(NULL, "summary", NULL, hc(cmd, "summary"), |
533 | ctx->qry.head, NULL, NULL); | 533 | ctx->qry.head, NULL, NULL); |
534 | cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, | 534 | cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, |
535 | ctx->qry.sha1, NULL); | 535 | ctx->qry.sha1, NULL); |
536 | cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, | 536 | cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, |
537 | NULL, NULL, 0, NULL, NULL); | 537 | NULL, NULL, 0, NULL, NULL); |
538 | cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, | 538 | cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, |
539 | ctx->qry.sha1, NULL); | 539 | ctx->qry.sha1, NULL); |
540 | cgit_commit_link("commit", NULL, hc(cmd, "commit"), | 540 | cgit_commit_link("commit", NULL, hc(cmd, "commit"), |
541 | ctx->qry.head, ctx->qry.sha1); | 541 | ctx->qry.head, ctx->qry.sha1); |
542 | cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, | 542 | cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, |
543 | ctx->qry.sha1, ctx->qry.sha2, NULL); | 543 | ctx->qry.sha1, ctx->qry.sha2, NULL); |
544 | cgit_patch_link("patch", NULL, hc(cmd, "patch"), ctx->qry.head, | 544 | cgit_patch_link("patch", NULL, hc(cmd, "patch"), ctx->qry.head, |
545 | ctx->qry.sha1); | 545 | ctx->qry.sha1); |
546 | html("</td><td class='form'>"); | 546 | html("</td><td class='form'>"); |
547 | html("<form class='right' method='get' action='"); | 547 | html("<form class='right' method='get' action='"); |
548 | if (ctx->cfg.virtual_root) | 548 | if (ctx->cfg.virtual_root) |
549 | html_attr(cgit_fileurl(ctx->qry.repo, "log", | 549 | html_attr(cgit_fileurl(ctx->qry.repo, "log", |
550 | ctx->qry.path, NULL)); | 550 | ctx->qry.path, NULL)); |
551 | html("'>\n"); | 551 | html("'>\n"); |
552 | add_hidden_formfields(1, 0, "log"); | 552 | add_hidden_formfields(1, 0, "log"); |
553 | html("<select name='qt'>\n"); | 553 | html("<select name='qt'>\n"); |
554 | html_option("grep", "log msg", ctx->qry.grep); | 554 | html_option("grep", "log msg", ctx->qry.grep); |
555 | html_option("author", "author", ctx->qry.grep); | 555 | html_option("author", "author", ctx->qry.grep); |
556 | html_option("committer", "committer", ctx->qry.grep); | 556 | html_option("committer", "committer", ctx->qry.grep); |
557 | html("</select>\n"); | 557 | html("</select>\n"); |
558 | html("<input class='txt' type='text' size='8' name='q' value='"); | 558 | html("<input class='txt' type='text' size='10' name='q' value='"); |
559 | html_attr(ctx->qry.search); | 559 | html_attr(ctx->qry.search); |
560 | html("'/>\n"); | 560 | html("'/>\n"); |
561 | html("<input type='submit' value='search'/>\n"); | 561 | html("<input type='submit' value='search'/>\n"); |
562 | html("</form>\n"); | 562 | html("</form>\n"); |
563 | } else { | 563 | } else { |
564 | html("<a class='active' href='"); | 564 | html("<a class='active' href='"); |
565 | html_attr(cgit_rooturl()); | 565 | html_attr(cgit_rooturl()); |
566 | html("'>index</a>\n"); | 566 | html("'>index</a>\n"); |
567 | html("</td><td class='form'>"); | ||
568 | html("<form method='get' action='"); | ||
569 | html_attr(cgit_rooturl()); | ||
570 | html("'>\n"); | ||
571 | html("<input type='text' name='q' size='10' value='"); | ||
572 | html_attr(ctx->qry.search); | ||
573 | html("'/>\n"); | ||
574 | html("<input type='submit' value='search'/>\n"); | ||
575 | html("</form>"); | ||
567 | } | 576 | } |
568 | html("</td></tr></table>\n"); | 577 | html("</td></tr></table>\n"); |
569 | html("<div class='content'>"); | 578 | html("<div class='content'>"); |
570 | } | 579 | } |
571 | 580 | ||
572 | void cgit_print_filemode(unsigned short mode) | 581 | void cgit_print_filemode(unsigned short mode) |
573 | { | 582 | { |
574 | if (S_ISDIR(mode)) | 583 | if (S_ISDIR(mode)) |
575 | html("d"); | 584 | html("d"); |
576 | else if (S_ISLNK(mode)) | 585 | else if (S_ISLNK(mode)) |
577 | html("l"); | 586 | html("l"); |
578 | else if (S_ISGITLINK(mode)) | 587 | else if (S_ISGITLINK(mode)) |
579 | html("m"); | 588 | html("m"); |
580 | else | 589 | else |
581 | html("-"); | 590 | html("-"); |
582 | html_fileperm(mode >> 6); | 591 | html_fileperm(mode >> 6); |
583 | html_fileperm(mode >> 3); | 592 | html_fileperm(mode >> 3); |
584 | html_fileperm(mode); | 593 | html_fileperm(mode); |
585 | } | 594 | } |
586 | 595 | ||
587 | void cgit_print_snapshot_links(const char *repo, const char *head, | 596 | void cgit_print_snapshot_links(const char *repo, const char *head, |
588 | const char *hex, int snapshots) | 597 | const char *hex, int snapshots) |
589 | { | 598 | { |
590 | const struct cgit_snapshot_format* f; | 599 | const struct cgit_snapshot_format* f; |
591 | char *filename; | 600 | char *filename; |
592 | 601 | ||
593 | for (f = cgit_snapshot_formats; f->suffix; f++) { | 602 | for (f = cgit_snapshot_formats; f->suffix; f++) { |
594 | if (!(snapshots & f->bit)) | 603 | if (!(snapshots & f->bit)) |
595 | continue; | 604 | continue; |
596 | filename = fmt("%s-%s%s", cgit_repobasename(repo), hex, | 605 | filename = fmt("%s-%s%s", cgit_repobasename(repo), hex, |
597 | f->suffix); | 606 | f->suffix); |
598 | cgit_snapshot_link(filename, NULL, NULL, (char *)head, | 607 | cgit_snapshot_link(filename, NULL, NULL, (char *)head, |