| 15 |
***************************************************************************/ |
***************************************************************************/ |
| 16 |
|
|
| 17 |
#include "article_cache.h" |
#include "article_cache.h" |
| 18 |
|
#include "article_op.h" |
| 19 |
#include "article_post.h" |
#include "article_post.h" |
| 20 |
#include "article_view_log.h" |
#include "article_view_log.h" |
| 21 |
#include "article_del.h" |
#include "article_del.h" |
| 33 |
#include <time.h> |
#include <time.h> |
| 34 |
#include <sys/param.h> |
#include <sys/param.h> |
| 35 |
|
|
| 36 |
|
static int section_aid_locations[BBS_max_section] = {0}; |
| 37 |
static int section_topic_view_mode = 0; |
static int section_topic_view_mode = 0; |
| 38 |
static int section_topic_view_tid = -1; |
static int section_topic_view_tid = -1; |
| 39 |
|
|
| 40 |
enum select_cmd_t |
enum select_cmd_t |
| 41 |
{ |
{ |
| 42 |
EXIT_SECTION = 0, |
EXIT_SECTION = 0, |
| 43 |
VIEW_ARTICLE = 1, |
VIEW_ARTICLE, |
| 44 |
CHANGE_PAGE = 2, |
CHANGE_PAGE, |
| 45 |
SHOW_HELP = 3, |
SHOW_HELP, |
| 46 |
CHANGE_NAME_DISPLAY = 4, |
CHANGE_NAME_DISPLAY, |
| 47 |
POST_ARTICLE = 5, |
POST_ARTICLE, |
| 48 |
EDIT_ARTICLE = 6, |
EDIT_ARTICLE, |
| 49 |
DELETE_ARTICLE = 7, |
DELETE_ARTICLE, |
| 50 |
FIRST_TOPIC_ARTICLE = 8, |
QUERY_ARTICLE, |
| 51 |
LAST_TOPIC_ARTICLE = 9, |
FIRST_TOPIC_ARTICLE, |
| 52 |
VIEW_EX_DIR = 10, |
LAST_TOPIC_ARTICLE, |
| 53 |
|
SCAN_NEW_ARTICLE, |
| 54 |
|
VIEW_EX_DIR, |
| 55 |
}; |
}; |
| 56 |
|
|
| 57 |
static int section_list_draw_items(int page_id, ARTICLE *p_articles[], int article_count, int display_nickname, int ontop_start_offset) |
static int section_list_draw_items(int page_id, ARTICLE *p_articles[], int article_count, int display_nickname, int ontop_start_offset) |
| 63 |
int eol; |
int eol; |
| 64 |
int len; |
int len; |
| 65 |
int i; |
int i; |
| 66 |
|
size_t j; |
| 67 |
char article_flag; |
char article_flag; |
| 68 |
int is_viewed; |
int is_viewed; |
| 69 |
time_t tm_now; |
time_t tm_now; |
| 114 |
strncpy(title_f, (p_articles[i]->tid == 0 ? "● " : ""), sizeof(title_f) - 1); |
strncpy(title_f, (p_articles[i]->tid == 0 ? "● " : ""), sizeof(title_f) - 1); |
| 115 |
title_f[sizeof(title_f) - 1] = '\0'; |
title_f[sizeof(title_f) - 1] = '\0'; |
| 116 |
strncat(title_f, (p_articles[i]->transship ? "[转载]" : ""), sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f))); |
strncat(title_f, (p_articles[i]->transship ? "[转载]" : ""), sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f))); |
| 117 |
strncat(title_f, p_articles[i]->title, sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f))); |
|
| 118 |
|
// Rewrite title with "Re: Re: " prefix into "Re: ... " |
| 119 |
|
j = 0; |
| 120 |
|
if (p_articles[i]->tid != 0) |
| 121 |
|
{ |
| 122 |
|
while (strncmp(p_articles[i]->title + j, "Re: ", strlen("Re: ")) == 0) |
| 123 |
|
{ |
| 124 |
|
j += strlen("Re: "); |
| 125 |
|
} |
| 126 |
|
if (j >= strlen("Re: Re: ")) |
| 127 |
|
{ |
| 128 |
|
strncat(title_f, "Re: ... ", sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f))); |
| 129 |
|
} |
| 130 |
|
else |
| 131 |
|
{ |
| 132 |
|
j = 0; |
| 133 |
|
} |
| 134 |
|
} |
| 135 |
|
strncat(title_f, p_articles[i]->title + j, sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f))); |
| 136 |
|
|
| 137 |
len = split_line(title_f, 47 - (display_nickname ? 8 : 0), &eol, &title_f_len, 1); |
len = split_line(title_f, 47 - (display_nickname ? 8 : 0), &eol, &title_f_len, 1); |
| 138 |
if (title_f[len] != '\0') |
if (title_f[len] != '\0') |
| 139 |
{ |
{ |
| 146 |
prints(" \033[1;33m[提示]\033[m %c %s%*s %s %s%s\033[m", |
prints(" \033[1;33m[提示]\033[m %c %s%*s %s %s%s\033[m", |
| 147 |
article_flag, |
article_flag, |
| 148 |
(display_nickname ? p_articles[i]->nickname : p_articles[i]->username), |
(display_nickname ? p_articles[i]->nickname : p_articles[i]->username), |
| 149 |
(display_nickname ? BBS_nickname_max_len - (int)strnlen(p_articles[i]->nickname, sizeof(p_articles[i]->nickname)) |
(display_nickname ? BBS_nickname_max_len / 2 - str_length(p_articles[i]->nickname, 1) |
| 150 |
: BBS_username_max_len - (int)strnlen(p_articles[i]->username, sizeof(p_articles[i]->username))), |
: BBS_username_max_len - str_length(p_articles[i]->username, 1)), |
| 151 |
"", |
"", |
| 152 |
str_time, |
str_time, |
| 153 |
(p_articles[i]->aid == section_topic_view_tid |
(p_articles[i]->aid == section_topic_view_tid |
| 168 |
p_articles[i]->aid, |
p_articles[i]->aid, |
| 169 |
article_flag, |
article_flag, |
| 170 |
(display_nickname ? p_articles[i]->nickname : p_articles[i]->username), |
(display_nickname ? p_articles[i]->nickname : p_articles[i]->username), |
| 171 |
(display_nickname ? BBS_nickname_max_len - (int)strnlen(p_articles[i]->nickname, sizeof(p_articles[i]->nickname)) |
(display_nickname ? BBS_nickname_max_len / 2 - str_length(p_articles[i]->nickname, 1) |
| 172 |
: BBS_username_max_len - (int)strnlen(p_articles[i]->username, sizeof(p_articles[i]->username))), |
: BBS_username_max_len - str_length(p_articles[i]->username, 1)), |
| 173 |
"", |
"", |
| 174 |
str_time, |
str_time, |
| 175 |
(p_articles[i]->aid == section_topic_view_tid |
(p_articles[i]->aid == section_topic_view_tid |
| 262 |
case Ctrl('P'): |
case Ctrl('P'): |
| 263 |
return POST_ARTICLE; |
return POST_ARTICLE; |
| 264 |
case 'E': |
case 'E': |
| 265 |
return EDIT_ARTICLE; |
if (item_count > 0) |
| 266 |
|
{ |
| 267 |
|
return EDIT_ARTICLE; |
| 268 |
|
} |
| 269 |
|
break; |
| 270 |
case 'd': |
case 'd': |
| 271 |
return DELETE_ARTICLE; |
if (item_count > 0) |
| 272 |
|
{ |
| 273 |
|
return DELETE_ARTICLE; |
| 274 |
|
} |
| 275 |
|
break; |
| 276 |
|
case Ctrl('Q'): |
| 277 |
|
if (item_count > 0) |
| 278 |
|
{ |
| 279 |
|
return QUERY_ARTICLE; |
| 280 |
|
} |
| 281 |
|
break; |
| 282 |
case KEY_HOME: |
case KEY_HOME: |
| 283 |
*p_page_id = 0; |
*p_page_id = 0; |
| 284 |
case 'P': |
case 'P': |
| 293 |
(*p_page_id)--; |
(*p_page_id)--; |
| 294 |
*p_selected_index = BBS_article_limit_per_page - 1; |
*p_selected_index = BBS_article_limit_per_page - 1; |
| 295 |
} |
} |
| 296 |
|
else if (ch == KEY_UP || ch == 'k') // Rotate to the tail of section list |
| 297 |
|
{ |
| 298 |
|
if (total_page > 0) |
| 299 |
|
{ |
| 300 |
|
*p_page_id = total_page - 1; |
| 301 |
|
} |
| 302 |
|
if (item_count > 0) |
| 303 |
|
{ |
| 304 |
|
*p_selected_index = item_count - 1; |
| 305 |
|
} |
| 306 |
|
} |
| 307 |
} |
} |
| 308 |
else |
else |
| 309 |
{ |
{ |
| 331 |
(*p_page_id)++; |
(*p_page_id)++; |
| 332 |
*p_selected_index = 0; |
*p_selected_index = 0; |
| 333 |
} |
} |
| 334 |
else // end of last page |
else if (ch == KEY_DOWN || ch == 'j') // Rotate to the head of section list |
| 335 |
{ |
{ |
| 336 |
return CHANGE_PAGE; // force refresh pages |
*p_page_id = 0; |
| 337 |
|
*p_selected_index = 0; |
| 338 |
} |
} |
| 339 |
} |
} |
| 340 |
else |
else |
| 343 |
} |
} |
| 344 |
break; |
break; |
| 345 |
case '=': |
case '=': |
| 346 |
return FIRST_TOPIC_ARTICLE; |
if (item_count > 0) |
| 347 |
|
{ |
| 348 |
|
return FIRST_TOPIC_ARTICLE; |
| 349 |
|
} |
| 350 |
|
break; |
| 351 |
case '\\': |
case '\\': |
| 352 |
return LAST_TOPIC_ARTICLE; |
if (item_count > 0) |
| 353 |
|
{ |
| 354 |
|
return LAST_TOPIC_ARTICLE; |
| 355 |
|
} |
| 356 |
|
break; |
| 357 |
|
case 'S': |
| 358 |
|
if (item_count > 0) |
| 359 |
|
{ |
| 360 |
|
return SCAN_NEW_ARTICLE; |
| 361 |
|
} |
| 362 |
|
break; |
| 363 |
case 'h': |
case 'h': |
| 364 |
return SHOW_HELP; |
return SHOW_HELP; |
| 365 |
case 'x': |
case 'x': |
| 486 |
return 0; |
return 0; |
| 487 |
} |
} |
| 488 |
|
|
| 489 |
int section_list_display(const char *sname) |
int section_list_display(const char *sname, int32_t aid) |
| 490 |
{ |
{ |
| 491 |
static int display_nickname = 0; |
static int display_nickname = 0; |
| 492 |
|
|
| 493 |
SECTION_LIST *p_section; |
SECTION_LIST *p_section; |
| 494 |
|
int64_t section_index; |
| 495 |
|
int32_t aid_location; |
| 496 |
char stitle[BBS_section_title_max_len + 1]; |
char stitle[BBS_section_title_max_len + 1]; |
| 497 |
char master_list[(BBS_username_max_len + 1) * 3 + 1]; |
char master_list[(BBS_username_max_len + 1) * 3 + 1]; |
| 498 |
char page_info_str[LINE_BUFFER_LEN]; |
char page_info_str[LINE_BUFFER_LEN]; |
| 508 |
int direction; |
int direction; |
| 509 |
ARTICLE article_new; |
ARTICLE article_new; |
| 510 |
int page_id_cur; |
int page_id_cur; |
| 511 |
|
const ARTICLE *p_article_locate; |
| 512 |
|
|
| 513 |
p_section = section_list_find_by_name(sname); |
p_section = section_list_find_by_name(sname, §ion_index); |
| 514 |
if (p_section == NULL) |
if (p_section == NULL) |
| 515 |
{ |
{ |
| 516 |
log_error("Section %s not found\n", sname); |
log_error("Section %s not found\n", sname); |
| 534 |
return -2; |
return -2; |
| 535 |
} |
} |
| 536 |
|
|
| 537 |
|
if (aid == 0) |
| 538 |
|
{ |
| 539 |
|
aid_location = section_aid_locations[section_index]; |
| 540 |
|
} |
| 541 |
|
else |
| 542 |
|
{ |
| 543 |
|
aid_location = aid; |
| 544 |
|
} |
| 545 |
|
|
| 546 |
|
// Locate at article with aid_locate |
| 547 |
|
if (aid_location > 0) |
| 548 |
|
{ |
| 549 |
|
p_article_locate = article_block_find_by_aid(aid_location); |
| 550 |
|
if (p_article_locate == NULL) |
| 551 |
|
{ |
| 552 |
|
log_error("article_block_find_by_aid(%d) error\n", aid_location); |
| 553 |
|
return -3; |
| 554 |
|
} |
| 555 |
|
|
| 556 |
|
ret = locate_article_in_section(p_section, p_article_locate, 0, 0, |
| 557 |
|
&page_id, &selected_index, &article_count); |
| 558 |
|
if (ret < 0) |
| 559 |
|
{ |
| 560 |
|
log_error("locate_article_in_section(sid=%d, aid=%d, direction=0, step=0) error\n", |
| 561 |
|
p_section->sid, p_article_locate->aid); |
| 562 |
|
return -3; |
| 563 |
|
} |
| 564 |
|
} |
| 565 |
|
|
| 566 |
if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0) |
if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0) |
| 567 |
{ |
{ |
| 568 |
log_error("section_list_draw_screen() error\n"); |
log_error("section_list_draw_screen() error\n"); |
| 576 |
return -3; |
return -3; |
| 577 |
} |
} |
| 578 |
|
|
| 579 |
|
section_topic_view_tid = -1; |
| 580 |
|
|
| 581 |
if (article_count == 0) // empty section |
if (article_count == 0) // empty section |
| 582 |
{ |
{ |
| 583 |
selected_index = 0; |
selected_index = 0; |
| 584 |
} |
} |
| 585 |
|
else if (aid > 0) |
| 586 |
|
{ |
| 587 |
|
// Update current topic |
| 588 |
|
section_topic_view_tid = (p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid); |
| 589 |
|
|
| 590 |
|
// Update current aid location |
| 591 |
|
section_aid_locations[section_index] = p_articles[selected_index]->aid; |
| 592 |
|
} |
| 593 |
|
|
| 594 |
while (!SYS_server_exit) |
while (!SYS_server_exit) |
| 595 |
{ |
{ |
| 794 |
// Update current topic |
// Update current topic |
| 795 |
section_topic_view_tid = (p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid); |
section_topic_view_tid = (p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid); |
| 796 |
|
|
| 797 |
|
// Update current aid location |
| 798 |
|
section_aid_locations[section_index] = p_articles[selected_index]->aid; |
| 799 |
|
|
| 800 |
if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0) |
if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0) |
| 801 |
{ |
{ |
| 802 |
log_error("section_list_draw_screen() error\n"); |
log_error("section_list_draw_screen() error\n"); |
| 883 |
return -2; |
return -2; |
| 884 |
} |
} |
| 885 |
break; |
break; |
| 886 |
|
case QUERY_ARTICLE: |
| 887 |
|
if ((ret = display_article_meta(p_articles[selected_index]->aid)) < 0) |
| 888 |
|
{ |
| 889 |
|
log_error("display_article_meta(aid=%d) error\n", p_articles[selected_index]->aid); |
| 890 |
|
} |
| 891 |
|
if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0) |
| 892 |
|
{ |
| 893 |
|
log_error("section_list_draw_screen() error\n"); |
| 894 |
|
return -2; |
| 895 |
|
} |
| 896 |
|
break; |
| 897 |
case FIRST_TOPIC_ARTICLE: |
case FIRST_TOPIC_ARTICLE: |
| 898 |
case LAST_TOPIC_ARTICLE: |
case LAST_TOPIC_ARTICLE: |
| 899 |
page_id_cur = page_id; |
page_id_cur = page_id; |
| 907 |
return -3; |
return -3; |
| 908 |
} |
} |
| 909 |
else if (ret > 0 && page_id != page_id_cur) // found and page changed |
else if (ret > 0 && page_id != page_id_cur) // found and page changed |
| 910 |
|
{ |
| 911 |
|
ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset); |
| 912 |
|
if (ret < 0) |
| 913 |
|
{ |
| 914 |
|
log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id); |
| 915 |
|
return -3; |
| 916 |
|
} |
| 917 |
|
} |
| 918 |
|
break; |
| 919 |
|
case SCAN_NEW_ARTICLE: |
| 920 |
|
ret = scan_unread_article_in_section(p_section, p_articles[selected_index], &p_article_locate); |
| 921 |
|
if (ret < 0) |
| 922 |
|
{ |
| 923 |
|
log_error("scan_unread_article_in_section(sid=%d, aid=%d) error\n", |
| 924 |
|
p_section->sid, p_articles[selected_index]->aid); |
| 925 |
|
return -3; |
| 926 |
|
} |
| 927 |
|
else if (ret == 0) // not found |
| 928 |
|
{ |
| 929 |
|
break; |
| 930 |
|
} |
| 931 |
|
page_id_cur = page_id; |
| 932 |
|
ret = locate_article_in_section(p_section, p_article_locate, 0, 0, |
| 933 |
|
&page_id, &selected_index, &article_count); |
| 934 |
|
if (ret < 0) |
| 935 |
|
{ |
| 936 |
|
log_error("locate_article_in_section(sid=%d, aid=%d, direction=0, step=0) error\n", |
| 937 |
|
p_section->sid, p_article_locate->aid); |
| 938 |
|
return -3; |
| 939 |
|
} |
| 940 |
|
else if (ret > 0 && page_id != page_id_cur) // found and page changed |
| 941 |
{ |
{ |
| 942 |
ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset); |
ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset); |
| 943 |
if (ret < 0) |
if (ret < 0) |