/[LeafOK_CVS]/lbbs/src/section_list_display.c
ViewVC logotype

Contents of /lbbs/src/section_list_display.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.59 - (show annotations)
Sun Nov 2 08:13:50 2025 UTC (4 months, 1 week ago) by sysadm
Branch: MAIN
Changes since 1.58: +114 -1 lines
Content type: text/x-csrc
Add section_aid_locations_[load|save]

1 /***************************************************************************
2 section_list_display.c - description
3 -------------------
4 Copyright : (C) 2004-2025 by Leaflet
5 Email : leaflet@leafok.com
6 ***************************************************************************/
7
8 /***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 3 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16
17 #include "article_cache.h"
18 #include "article_favor.h"
19 #include "article_op.h"
20 #include "article_post.h"
21 #include "article_view_log.h"
22 #include "article_del.h"
23 #include "common.h"
24 #include "io.h"
25 #include "log.h"
26 #include "login.h"
27 #include "menu.h"
28 #include "menu_proc.h"
29 #include "section_list_display.h"
30 #include "section_list_loader.h"
31 #include "screen.h"
32 #include "str_process.h"
33 #include "user_info_display.h"
34 #include "user_priv.h"
35 #include <errno.h>
36 #include <string.h>
37 #include <time.h>
38 #include <sys/param.h>
39
40 static int32_t section_aid_locations[BBS_max_section] = {0};
41 static int section_topic_view_mode = 0;
42 static int section_topic_view_tid = -1;
43
44 enum select_cmd_t
45 {
46 EXIT_SECTION = 0,
47 VIEW_ARTICLE,
48 CHANGE_PAGE,
49 SHOW_HELP,
50 CHANGE_NAME_DISPLAY,
51 POST_ARTICLE,
52 EDIT_ARTICLE,
53 DELETE_ARTICLE,
54 QUERY_ARTICLE,
55 QUERY_USER,
56 SET_FAVOR_ARTICLE,
57 UNSET_FAVOR_ARTICLE,
58 FIRST_TOPIC_ARTICLE,
59 LAST_TOPIC_ARTICLE,
60 SCAN_NEW_ARTICLE,
61 VIEW_EX_DIR,
62 SHOW_TOP10,
63 };
64
65 static int section_list_draw_items(int page_id, ARTICLE *p_articles[], int article_count, int display_nickname, int ontop_start_offset)
66 {
67 char str_time[LINE_BUFFER_LEN];
68 struct tm tm_sub;
69 char title_f[BBS_article_title_max_len + 1];
70 int title_f_len;
71 int eol;
72 int len;
73 int i;
74 size_t j;
75 char article_flag;
76 int is_viewed;
77 int is_favor;
78 time_t tm_now;
79
80 time(&tm_now);
81
82 clrline(4, 23);
83
84 for (i = 0; i < article_count; i++)
85 {
86 if (p_articles[i]->uid == BBS_priv.uid)
87 {
88 is_viewed = 1;
89 }
90 else
91 {
92 is_viewed = article_view_log_is_viewed(p_articles[i]->aid, &BBS_article_view_log);
93 if (is_viewed < 0)
94 {
95 log_error("article_view_log_is_viewed(aid=%d) error\n", p_articles[i]->aid);
96 is_viewed = 0;
97 }
98 }
99
100 if (p_articles[i]->tid == 0)
101 {
102 is_favor = article_favor_check(p_articles[i]->aid, &BBS_article_favor);
103 if (is_favor < 0)
104 {
105 log_error("article_favor_check(aid=%d) error\n", p_articles[i]->aid);
106 is_favor = 0;
107 }
108 }
109 else
110 {
111 is_favor = 0;
112 }
113
114 if (p_articles[i]->excerption)
115 {
116 article_flag = (is_viewed ? 'm' : 'M');
117 }
118 else if (p_articles[i]->lock && is_viewed)
119 {
120 article_flag = 'x';
121 }
122 else
123 {
124 article_flag = (is_viewed ? ' ' : 'N');
125 }
126
127 localtime_r(&p_articles[i]->sub_dt, &tm_sub);
128 if (tm_now - p_articles[i]->sub_dt < 3600 * 24 * 365)
129 {
130 strftime(str_time, sizeof(str_time), "%b %e ", &tm_sub);
131 }
132 else
133 {
134 strftime(str_time, sizeof(str_time), "%m/%Y", &tm_sub);
135 }
136
137 strncpy(title_f, (p_articles[i]->tid == 0 ? "● " : ""), sizeof(title_f) - 1);
138 title_f[sizeof(title_f) - 1] = '\0';
139 strncat(title_f, (p_articles[i]->transship ? "[转载]" : ""), sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f)));
140
141 // Rewrite title with "Re: Re: " prefix into "Re: ... "
142 j = 0;
143 if (p_articles[i]->tid != 0)
144 {
145 while (strncmp(p_articles[i]->title + j, "Re: ", strlen("Re: ")) == 0)
146 {
147 j += strlen("Re: ");
148 }
149 if (j >= strlen("Re: Re: "))
150 {
151 strncat(title_f, "Re: ... ", sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f)));
152 }
153 else
154 {
155 j = 0;
156 }
157 }
158 strncat(title_f, p_articles[i]->title + j, sizeof(title_f) - 1 - strnlen(title_f, sizeof(title_f)));
159
160 len = split_line(title_f, 59 - (display_nickname ? BBS_nickname_max_len / 2 : BBS_username_max_len), &eol, &title_f_len, 1);
161 if (title_f[len] != '\0')
162 {
163 title_f[len] = '\0';
164 }
165
166 moveto(4 + i, 1);
167 if (i >= ontop_start_offset)
168 {
169 prints(" \033[1;33m[提示]\033[m%c%c %s%*s %s %s%s\033[m",
170 (is_favor ? '@' : ' '),
171 article_flag,
172 (display_nickname ? p_articles[i]->nickname : p_articles[i]->username),
173 (display_nickname ? BBS_nickname_max_len / 2 - str_length(p_articles[i]->nickname, 1)
174 : BBS_username_max_len - str_length(p_articles[i]->username, 1)),
175 "",
176 str_time,
177 (p_articles[i]->aid == section_topic_view_tid
178 ? "\033[1;33m"
179 : (p_articles[i]->tid == section_topic_view_tid
180 ? "\033[1;36m"
181 : "")),
182 title_f);
183 }
184 else
185 {
186 prints(" %s%7d\033[m%c%c %s%*s %s %s%s\033[m",
187 (p_articles[i]->aid == section_topic_view_tid
188 ? "\033[1;33m"
189 : (p_articles[i]->tid == section_topic_view_tid
190 ? "\033[1;36m"
191 : "")),
192 p_articles[i]->aid,
193 (is_favor ? '@' : ' '),
194 article_flag,
195 (display_nickname ? p_articles[i]->nickname : p_articles[i]->username),
196 (display_nickname ? BBS_nickname_max_len / 2 - str_length(p_articles[i]->nickname, 1)
197 : BBS_username_max_len - str_length(p_articles[i]->username, 1)),
198 "",
199 str_time,
200 (p_articles[i]->aid == section_topic_view_tid
201 ? "\033[1;33m"
202 : (p_articles[i]->tid == section_topic_view_tid
203 ? "\033[1;36m"
204 : "")),
205 title_f);
206 }
207 }
208
209 return 0;
210 }
211
212 static int section_list_draw_screen(const char *sname, const char *stitle, const char *master_list, int display_nickname)
213 {
214 char str_section_master[LINE_BUFFER_LEN] = "诚征版主中";
215 char str_section_name[LINE_BUFFER_LEN];
216
217 if (master_list[0] != '\0')
218 {
219 snprintf(str_section_master, sizeof(str_section_master), "版主:%s", master_list);
220 }
221 snprintf(str_section_name, sizeof(str_section_name), "讨论区 [%s]", sname);
222
223 clearscr();
224 show_top(str_section_master, stitle, str_section_name);
225 moveto(2, 0);
226 prints("返回[\033[1;32m←\033[0;37m,\033[1;32mESC\033[0;37m] 选择[\033[1;32m↑\033[0;37m,\033[1;32m↓\033[0;37m] "
227 "阅读[\033[1;32m→\033[0;37m,\033[1;32mENTER\033[0;37m] 发表[\033[1;32mCtrl-P\033[0;37m] "
228 "%s[\033[1;32mn\033[0;37m] 精华区[\033[1;32mx\033[0;37m] 帮助[\033[1;32mh\033[0;37m]\033[m",
229 (display_nickname ? "用户名" : "昵称"));
230 moveto(3, 0);
231 if (display_nickname)
232 {
233 prints("\033[44;37m \033[1;37m 编 号 发布者昵称 日 期 文章标题 \033[m");
234 }
235 else
236 {
237 prints("\033[44;37m \033[1;37m 编 号 发 布 者 日 期 文章标题 \033[m");
238 }
239
240 return 0;
241 }
242
243 static enum select_cmd_t section_list_select(int total_page, int item_count, int *p_page_id, int *p_selected_index)
244 {
245 int old_page_id = *p_page_id;
246 int old_selected_index = *p_selected_index;
247 int ch;
248 time_t last_refresh_tm = time(NULL);
249
250 if (item_count > 0 && *p_selected_index >= 0)
251 {
252 moveto(4 + *p_selected_index, 1);
253 outc('>');
254 iflush();
255 }
256
257 while (!SYS_server_exit)
258 {
259 ch = igetch(100);
260
261 if (ch != KEY_NULL && ch != KEY_TIMEOUT)
262 {
263 BBS_last_access_tm = time(NULL);
264 }
265
266 switch (ch)
267 {
268 case KEY_NULL: // broken pipe
269 log_error("KEY_NULL\n");
270 return EXIT_SECTION;
271 case KEY_TIMEOUT:
272 if (time(NULL) - BBS_last_access_tm >= MAX_DELAY_TIME)
273 {
274 log_error("User input timeout\n");
275 return EXIT_SECTION;
276 }
277 continue;
278 case KEY_ESC:
279 case KEY_LEFT:
280 return EXIT_SECTION;
281 case 'n':
282 return CHANGE_NAME_DISPLAY;
283 case CR:
284 case 'r':
285 case KEY_RIGHT:
286 if (item_count > 0)
287 {
288 return VIEW_ARTICLE;
289 }
290 break;
291 case Ctrl('P'):
292 return POST_ARTICLE;
293 case 'E':
294 if (item_count > 0)
295 {
296 return EDIT_ARTICLE;
297 }
298 break;
299 case 'd':
300 if (item_count > 0)
301 {
302 return DELETE_ARTICLE;
303 }
304 break;
305 case Ctrl('Q'):
306 if (item_count > 0)
307 {
308 return QUERY_ARTICLE;
309 }
310 break;
311 case Ctrl('A'):
312 if (item_count > 0)
313 {
314 return QUERY_USER;
315 }
316 break;
317 case 'F':
318 if (item_count > 0)
319 {
320 return SET_FAVOR_ARTICLE;
321 }
322 break;
323 case '-':
324 if (item_count > 0)
325 {
326 return UNSET_FAVOR_ARTICLE;
327 }
328 break;
329 case KEY_HOME:
330 *p_page_id = 0;
331 case 'P':
332 case KEY_PGUP:
333 *p_selected_index = 0;
334 case 'k':
335 case KEY_UP:
336 if (*p_selected_index <= 0)
337 {
338 if (*p_page_id > 0)
339 {
340 (*p_page_id)--;
341 *p_selected_index = BBS_article_limit_per_page - 1;
342 }
343 else if (ch == KEY_UP || ch == 'k') // Rotate to the tail of section list
344 {
345 if (total_page > 0)
346 {
347 *p_page_id = total_page - 1;
348 }
349 if (item_count > 0)
350 {
351 *p_selected_index = item_count - 1;
352 }
353 }
354 }
355 else
356 {
357 (*p_selected_index)--;
358 }
359 break;
360 case '$':
361 case KEY_END:
362 if (total_page > 0)
363 {
364 *p_page_id = total_page - 1;
365 }
366 case 'N':
367 case KEY_PGDN:
368 if (item_count > 0)
369 {
370 *p_selected_index = item_count - 1;
371 }
372 case 'j':
373 case KEY_DOWN:
374 if (*p_selected_index + 1 >= item_count) // next page
375 {
376 if (*p_page_id + 1 < total_page)
377 {
378 (*p_page_id)++;
379 *p_selected_index = 0;
380 }
381 else if (ch == KEY_DOWN || ch == 'j') // Rotate to the head of section list
382 {
383 *p_page_id = 0;
384 *p_selected_index = 0;
385 }
386 }
387 else
388 {
389 (*p_selected_index)++;
390 }
391 break;
392 case '=':
393 if (item_count > 0)
394 {
395 return FIRST_TOPIC_ARTICLE;
396 }
397 break;
398 case '\\':
399 if (item_count > 0)
400 {
401 return LAST_TOPIC_ARTICLE;
402 }
403 break;
404 case 'S':
405 if (item_count > 0)
406 {
407 return SCAN_NEW_ARTICLE;
408 }
409 break;
410 case 'h':
411 return SHOW_HELP;
412 case 'x':
413 return VIEW_EX_DIR;
414 case 'H':
415 return SHOW_TOP10;
416 default:
417 break;
418 }
419
420 if (old_page_id != *p_page_id)
421 {
422 return CHANGE_PAGE;
423 }
424
425 if (item_count > 0 && old_selected_index != *p_selected_index)
426 {
427 if (old_selected_index >= 0)
428 {
429 moveto(4 + old_selected_index, 1);
430 outc(' ');
431 }
432 if (*p_selected_index >= 0)
433 {
434 moveto(4 + *p_selected_index, 1);
435 outc('>');
436 }
437 iflush();
438
439 old_selected_index = *p_selected_index;
440 }
441
442 if (BBS_last_access_tm - last_refresh_tm >= BBS_section_list_load_interval)
443 {
444 return CHANGE_PAGE; // force section list refresh
445 }
446 }
447
448 return EXIT_SECTION;
449 }
450
451 static int display_article_key_handler(int *p_key, DISPLAY_CTX *p_ctx)
452 {
453 switch (*p_key)
454 {
455 case 'p':
456 case Ctrl('X'):
457 section_topic_view_mode = !section_topic_view_mode;
458 case 0: // Set msg
459 if (section_topic_view_mode)
460 {
461 snprintf(p_ctx->msg, sizeof(p_ctx->msg),
462 "| 返回[\033[32m←\033[33m,\033[32mESC\033[33m] "
463 "同主题阅读[\033[32m↑\033[33m/\033[32m↓\033[33m] "
464 "切换[\033[32mp\033[33m] 回复[\033[32mr\033[33m] 帮助[\033[32mh\033[33m] |");
465 }
466 else
467 {
468 snprintf(p_ctx->msg, sizeof(p_ctx->msg),
469 "| 返回[\033[32m←\033[33m,\033[32mESC\033[33m] "
470 "移动[\033[32m↑\033[33m/\033[32m↓\033[33m/\033[32mPgUp\033[33m/\033[32mPgDn\033[33m] "
471 "切换[\033[32mp\033[33m] 回复[\033[32mr\033[33m] 帮助[\033[32mh\033[33m] |");
472 }
473 *p_key = 0;
474 break;
475 case 'r': // Reply article
476 return 1;
477 case '=': // First topic article
478 return 1;
479 case '\\': // Last topic article
480 return 1;
481 case KEY_UP:
482 case KEY_PGUP:
483 case KEY_HOME:
484 if (p_ctx->reach_begin)
485 {
486 if (section_topic_view_mode)
487 {
488 *p_key = KEY_PGUP;
489 }
490 else
491 {
492 *p_key = KEY_UP;
493 }
494 return 1;
495 }
496 break;
497 case 'k':
498 if (section_topic_view_mode)
499 {
500 *p_key = KEY_PGUP;
501 }
502 else
503 {
504 *p_key = KEY_UP;
505 }
506 return 1;
507 case KEY_DOWN:
508 case KEY_PGDN:
509 case KEY_END:
510 if (p_ctx->reach_end)
511 {
512 if (section_topic_view_mode)
513 {
514 *p_key = KEY_PGDN;
515 }
516 else
517 {
518 *p_key = KEY_DOWN;
519 }
520 return 1;
521 }
522 break;
523 case 'j':
524 if (section_topic_view_mode)
525 {
526 *p_key = KEY_PGDN;
527 }
528 else
529 {
530 *p_key = KEY_DOWN;
531 }
532 return 1;
533 }
534
535 return 0;
536 }
537
538 int section_list_display(const char *sname, int32_t aid)
539 {
540 static int display_nickname = 0;
541
542 SECTION_LIST *p_section;
543 int64_t section_index;
544 int32_t aid_location;
545 char stitle[BBS_section_title_max_len + 1];
546 char master_list[(BBS_username_max_len + 1) * 3 + 1];
547 char page_info_str[LINE_BUFFER_LEN];
548 ARTICLE *p_articles[BBS_article_limit_per_page];
549 int article_count;
550 int page_count;
551 int ontop_start_offset;
552 int page_id = 0;
553 int selected_index = 0;
554 ARTICLE_CACHE cache;
555 int ret;
556 int loop;
557 int direction;
558 ARTICLE article_new;
559 int page_id_cur;
560 const ARTICLE *p_article_locate;
561 USER_INFO user_info;
562
563 p_section = section_list_find_by_name(sname);
564 if (p_section == NULL)
565 {
566 log_error("Section %s not found\n", sname);
567 return -1;
568 }
569
570 if (!checkpriv(&BBS_priv, p_section->sid, S_LIST))
571 {
572 log_error("Forbid access to unauthorized section, sid=%d, uid=%d\n",
573 p_section->sid, BBS_priv.uid);
574 return -1;
575 }
576
577 section_index = get_section_index(p_section);
578
579 if (get_section_info(p_section, NULL, stitle, master_list) < 0)
580 {
581 log_error("get_section_info(sid=%d) error\n", p_section->sid);
582 return -4;
583 }
584
585 if (aid == 0)
586 {
587 aid_location = section_aid_locations[section_index];
588 }
589 else
590 {
591 aid_location = aid;
592 }
593
594 // Locate at article with aid_locate
595 if (aid_location > 0)
596 {
597 p_article_locate = article_block_find_by_aid(aid_location);
598 if (p_article_locate == NULL)
599 {
600 log_error("article_block_find_by_aid(%d) error\n", aid_location);
601 return -3;
602 }
603
604 ret = locate_article_in_section(p_section, p_article_locate, 0, 0,
605 &page_id, &selected_index, &article_count);
606 if (ret < 0)
607 {
608 log_error("locate_article_in_section(sid=%d, aid=%d, direction=0, step=0) error\n",
609 p_section->sid, p_article_locate->aid);
610 return -3;
611 }
612 }
613
614 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
615 {
616 log_error("section_list_draw_screen() error\n");
617 return -2;
618 }
619
620 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
621 if (ret < 0)
622 {
623 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
624 return -3;
625 }
626
627 section_topic_view_tid = -1;
628
629 if (article_count == 0) // empty section
630 {
631 selected_index = 0;
632 }
633 else if (aid > 0)
634 {
635 // Update current topic
636 section_topic_view_tid = (p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid);
637
638 // Update current aid location
639 section_aid_locations[section_index] = p_articles[selected_index]->aid;
640 }
641
642 while (!SYS_server_exit)
643 {
644 ret = section_list_draw_items(page_id, p_articles, article_count, display_nickname, ontop_start_offset);
645 if (ret < 0)
646 {
647 log_error("section_list_draw_items(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
648 return -4;
649 }
650
651 snprintf(page_info_str, sizeof(page_info_str),
652 "\033[33m[第\033[36m%d\033[33m/\033[36m%d\033[33m页]",
653 page_id + 1, MAX(page_count, 1));
654
655 show_bottom(page_info_str);
656 iflush();
657
658 if (user_online_update(sname) < 0)
659 {
660 log_error("user_online_update(%s) error\n", sname);
661 }
662
663 ret = section_list_select(page_count, article_count, &page_id, &selected_index);
664 switch (ret)
665 {
666 case EXIT_SECTION:
667 return 0;
668 case CHANGE_PAGE:
669 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
670 if (ret < 0)
671 {
672 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
673 return -3;
674 }
675 if (article_count == 0) // empty section
676 {
677 selected_index = 0;
678 }
679 else if (selected_index >= article_count)
680 {
681 selected_index = article_count - 1;
682 }
683 break;
684 case VIEW_ARTICLE:
685 do
686 {
687 loop = 0;
688
689 if (article_cache_load(&cache, VAR_ARTICLE_CACHE_DIR, p_articles[selected_index]) < 0)
690 {
691 log_error("article_cache_load(aid=%d, cid=%d) error\n", p_articles[selected_index]->aid, p_articles[selected_index]->cid);
692 break;
693 }
694
695 if (user_online_update("VIEW_ARTICLE") < 0)
696 {
697 log_error("user_online_update(VIEW_ARTICLE) error\n");
698 }
699
700 ret = display_data(cache.p_data, cache.line_total, cache.line_offsets, 0,
701 display_article_key_handler, DATA_READ_HELP);
702
703 if (article_cache_unload(&cache) < 0)
704 {
705 log_error("article_cache_unload(aid=%d, cid=%d) error\n", p_articles[selected_index]->aid, p_articles[selected_index]->cid);
706 break;
707 }
708
709 // Update article_view_log
710 if (article_view_log_set_viewed(p_articles[selected_index]->aid, &BBS_article_view_log) < 0)
711 {
712 log_error("article_view_log_set_viewed(aid=%d) error\n", p_articles[selected_index]->aid);
713 }
714
715 switch (ret)
716 {
717 case KEY_UP:
718 if (selected_index <= 0)
719 {
720 if (page_id > 0)
721 {
722 page_id--;
723 selected_index = BBS_article_limit_per_page - 1;
724
725 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
726 if (ret < 0)
727 {
728 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
729 return -3;
730 }
731
732 if (article_count != BBS_article_limit_per_page) // page is not full
733 {
734 selected_index = MAX(0, article_count - 1);
735 }
736 else
737 {
738 loop = 1;
739 }
740 }
741 }
742 else
743 {
744 selected_index--;
745 loop = 1;
746 }
747 break;
748 case KEY_DOWN:
749 if (selected_index + 1 >= article_count) // next page
750 {
751 if (page_id + 1 < page_count)
752 {
753 page_id++;
754 selected_index = 0;
755
756 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
757 if (ret < 0)
758 {
759 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
760 return -3;
761 }
762
763 if (article_count == 0) // empty page
764 {
765 selected_index = 0;
766 }
767 else
768 {
769 loop = 1;
770 }
771 }
772 }
773 else
774 {
775 selected_index++;
776 loop = 1;
777 }
778 break;
779 case KEY_PGUP:
780 case KEY_PGDN:
781 direction = (ret == KEY_PGUP ? -1 : 1);
782 ret = locate_article_in_section(p_section, p_articles[selected_index], direction, 1,
783 &page_id, &selected_index, &article_count);
784 if (ret < 0)
785 {
786 log_error("locate_article_in_section(sid=%d, aid=%d, direction=%d, step=1) error\n",
787 p_section->sid, p_articles[selected_index]->aid, direction);
788 return -3;
789 }
790 else if (ret > 0) // found
791 {
792 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
793 if (ret < 0)
794 {
795 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
796 return -3;
797 }
798 loop = 1;
799 }
800 break;
801 case 'r': // Reply article
802 if (user_online_update("REPLY_ARTICLE") < 0)
803 {
804 log_error("user_online_update(REPLY_ARTICLE) error\n");
805 }
806
807 if (article_reply(p_section, p_articles[selected_index], &article_new) < 0)
808 {
809 log_error("article_reply(aid=%d) error\n", p_articles[selected_index]->aid);
810 }
811 loop = 1;
812 break;
813 case '=': // First topic article
814 case '\\': // Last topic article
815 page_id_cur = page_id;
816 direction = (ret == '=' ? -1 : 1);
817 ret = locate_article_in_section(p_section, p_articles[selected_index], direction, BBS_article_limit_per_section,
818 &page_id, &selected_index, &article_count);
819 if (ret < 0)
820 {
821 log_error("locate_article_in_section(sid=%d, aid=%d, direction=%d, step=%d) error\n",
822 p_section->sid, p_articles[selected_index]->aid, direction, BBS_article_limit_per_section);
823 return -3;
824 }
825 else if (ret > 0) // found
826 {
827 if (page_id != page_id_cur) // page changed
828 {
829 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
830 if (ret < 0)
831 {
832 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
833 return -3;
834 }
835 }
836 loop = 1;
837 }
838 break;
839 }
840 } while (loop);
841
842 // Update current topic
843 section_topic_view_tid = (p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid);
844
845 // Update current aid location
846 section_aid_locations[section_index] = p_articles[selected_index]->aid;
847
848 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
849 {
850 log_error("section_list_draw_screen() error\n");
851 return -2;
852 }
853 break;
854 case CHANGE_NAME_DISPLAY:
855 display_nickname = !display_nickname;
856 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
857 {
858 log_error("section_list_draw_screen() error\n");
859 return -2;
860 }
861 break;
862 case POST_ARTICLE:
863 if (user_online_update("POST_ARTICLE") < 0)
864 {
865 log_error("user_online_update(POST_ARTICLE) error\n");
866 }
867
868 if ((ret = article_post(p_section, &article_new)) < 0)
869 {
870 log_error("article_post(sid=%d) error\n", p_section->sid);
871 }
872 else if (ret > 0) // New article posted
873 {
874 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
875 if (ret < 0)
876 {
877 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
878 return -3;
879 }
880 }
881 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
882 {
883 log_error("section_list_draw_screen() error\n");
884 return -2;
885 }
886 break;
887 case EDIT_ARTICLE:
888 if (!checkpriv(&BBS_priv, p_section->sid, S_POST) ||
889 p_articles[selected_index]->uid != BBS_priv.uid)
890 {
891 break; // No permission
892 }
893
894 if (user_online_update("EDIT_ARTICLE") < 0)
895 {
896 log_error("user_online_update() error\n");
897 }
898
899 if (article_modify(p_section, p_articles[selected_index], &article_new) < 0)
900 {
901 log_error("article_modify(aid=%d) error\n", p_articles[selected_index]->aid);
902 }
903 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
904 {
905 log_error("section_list_draw_screen() error\n");
906 return -2;
907 }
908 break;
909 case DELETE_ARTICLE:
910 if (!checkpriv(&BBS_priv, p_section->sid, S_POST) ||
911 (!checkpriv(&BBS_priv, p_section->sid, S_MAN_S) && p_articles[selected_index]->uid != BBS_priv.uid))
912 {
913 break; // No permission
914 }
915 if ((ret = article_del(p_section, p_articles[selected_index])) < 0)
916 {
917 log_error("article_del(aid=%d) error\n", p_articles[selected_index]->aid);
918 }
919 else if (ret > 0) // Article deleted
920 {
921 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
922 if (ret < 0)
923 {
924 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
925 return -3;
926 }
927 }
928 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
929 {
930 log_error("section_list_draw_screen() error\n");
931 return -2;
932 }
933 break;
934 case QUERY_ARTICLE:
935 if ((ret = display_article_meta(p_articles[selected_index]->aid)) < 0)
936 {
937 log_error("display_article_meta(aid=%d) error\n", p_articles[selected_index]->aid);
938 }
939 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
940 {
941 log_error("section_list_draw_screen() error\n");
942 return -2;
943 }
944 break;
945 case QUERY_USER:
946 if ((ret = query_user_info_by_uid(p_articles[selected_index]->uid, &user_info)) < 0)
947 {
948 log_error("query_user_info_by_uid(uid=%d) error\n", p_articles[selected_index]->uid);
949 return -2;
950 }
951 else if (ret == 0)
952 {
953 clearscr();
954 prints("该用户已升天");
955 press_any_key();
956 }
957 else if (user_info_display(&user_info) < 0) // && ret > 0
958 {
959 log_error("user_info_display(uid=%d) error\n", p_articles[selected_index]->uid);
960 }
961
962 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
963 {
964 log_error("section_list_draw_screen() error\n");
965 return -2;
966 }
967 break;
968 case SET_FAVOR_ARTICLE:
969 ret = article_favor_set(p_articles[selected_index]->tid == 0
970 ? p_articles[selected_index]->aid
971 : p_articles[selected_index]->tid,
972 &BBS_article_favor, 1);
973 if (ret < 0)
974 {
975 log_error("article_favor_set(aid=%d, 1) error\n",
976 p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid);
977 }
978 break;
979 case UNSET_FAVOR_ARTICLE:
980 ret = article_favor_set(p_articles[selected_index]->tid == 0
981 ? p_articles[selected_index]->aid
982 : p_articles[selected_index]->tid,
983 &BBS_article_favor, 0);
984 if (ret < 0)
985 {
986 log_error("article_favor_set(aid=%d, 0) error\n",
987 p_articles[selected_index]->tid == 0 ? p_articles[selected_index]->aid : p_articles[selected_index]->tid);
988 }
989 break;
990 case FIRST_TOPIC_ARTICLE:
991 case LAST_TOPIC_ARTICLE:
992 page_id_cur = page_id;
993 direction = (ret == FIRST_TOPIC_ARTICLE ? -1 : 1);
994 ret = locate_article_in_section(p_section, p_articles[selected_index], direction, BBS_article_limit_per_section,
995 &page_id, &selected_index, &article_count);
996 if (ret < 0)
997 {
998 log_error("locate_article_in_section(sid=%d, aid=%d, direction=%d, step=%d) error\n",
999 p_section->sid, p_articles[selected_index]->aid, direction, BBS_article_limit_per_section);
1000 return -3;
1001 }
1002 else if (ret > 0 && page_id != page_id_cur) // found and page changed
1003 {
1004 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
1005 if (ret < 0)
1006 {
1007 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
1008 return -3;
1009 }
1010 }
1011 break;
1012 case SCAN_NEW_ARTICLE:
1013 ret = scan_unread_article_in_section(p_section, p_articles[selected_index], &p_article_locate);
1014 if (ret < 0)
1015 {
1016 log_error("scan_unread_article_in_section(sid=%d, aid=%d) error\n",
1017 p_section->sid, p_articles[selected_index]->aid);
1018 return -3;
1019 }
1020 else if (ret == 0) // not found
1021 {
1022 break;
1023 }
1024 page_id_cur = page_id;
1025 ret = locate_article_in_section(p_section, p_article_locate, 0, 0,
1026 &page_id, &selected_index, &article_count);
1027 if (ret < 0)
1028 {
1029 log_error("locate_article_in_section(sid=%d, aid=%d, direction=0, step=0) error\n",
1030 p_section->sid, p_article_locate->aid);
1031 return -3;
1032 }
1033 else if (ret > 0 && page_id != page_id_cur) // found and page changed
1034 {
1035 ret = query_section_articles(p_section, page_id, p_articles, &article_count, &page_count, &ontop_start_offset);
1036 if (ret < 0)
1037 {
1038 log_error("query_section_articles(sid=%d, page_id=%d) error\n", p_section->sid, page_id);
1039 return -3;
1040 }
1041 }
1042 break;
1043 case SHOW_HELP:
1044 // Display help information
1045 display_file(DATA_READ_HELP, 1);
1046 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
1047 {
1048 log_error("section_list_draw_screen() error\n");
1049 return -2;
1050 }
1051 break;
1052 case VIEW_EX_DIR:
1053 if (section_list_ex_dir_display(p_section) < 0)
1054 {
1055 log_error("section_list_ex_dir_display(sid=%d) error\n", p_section->sid);
1056 }
1057 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
1058 {
1059 log_error("section_list_draw_screen() error\n");
1060 return -2;
1061 }
1062 break;
1063 case SHOW_TOP10:
1064 show_top10_menu(NULL);
1065 if (section_list_draw_screen(sname, stitle, master_list, display_nickname) < 0)
1066 {
1067 log_error("section_list_draw_screen() error\n");
1068 return -2;
1069 }
1070 break;
1071 default:
1072 log_error("Unknown command %d\n", ret);
1073 }
1074 }
1075
1076 return 0;
1077 }
1078
1079 int section_list_ex_dir_display(SECTION_LIST *p_section)
1080 {
1081 MENU_SET ex_menu_set;
1082 int ch = 0;
1083
1084 if (p_section == NULL)
1085 {
1086 log_error("NULL pointer error\n");
1087 return -1;
1088 }
1089
1090 if (p_section->ex_menu_tm == 0) // N/A
1091 {
1092 moveto(2, 1);
1093 clrtoeol();
1094 prints("该版块精华区未开放");
1095 press_any_key();
1096 return 0;
1097 }
1098
1099 if (get_section_ex_menu_set(p_section, &ex_menu_set) < 0)
1100 {
1101 log_error("get_section_ex_menu_set(sid=%d) error\n", p_section->sid);
1102 return -3;
1103 }
1104 if (get_menu_shm_readonly(&ex_menu_set) < 0)
1105 {
1106 log_error("get_menu_shm_readonly(sid=%d) error\n", p_section->sid);
1107 return -3;
1108 }
1109
1110 clearscr();
1111 show_bottom("");
1112
1113 if (display_menu(&ex_menu_set) == 0)
1114 {
1115 while (!SYS_server_exit)
1116 {
1117 iflush();
1118 ch = igetch(100);
1119
1120 if (ch != KEY_NULL && ch != KEY_TIMEOUT)
1121 {
1122 BBS_last_access_tm = time(NULL);
1123 }
1124
1125 switch (ch)
1126 {
1127 case KEY_NULL: // broken pipe
1128 log_error("KEY_NULL\n");
1129 return 0;
1130 case KEY_TIMEOUT:
1131 if (time(NULL) - BBS_last_access_tm >= MAX_DELAY_TIME)
1132 {
1133 log_error("User input timeout\n");
1134 return 0;
1135 }
1136 continue;
1137 case CR:
1138 default:
1139 switch (menu_control(&ex_menu_set, ch))
1140 {
1141 case EXITMENU:
1142 ch = EXITMENU;
1143 break;
1144 case REDRAW:
1145 clearscr();
1146 show_bottom("");
1147 display_menu(&ex_menu_set);
1148 break;
1149 case NOREDRAW:
1150 case UNKNOWN_CMD:
1151 default:
1152 break;
1153 }
1154 }
1155
1156 if (ch == EXITMENU)
1157 {
1158 break;
1159 }
1160 }
1161 }
1162
1163 detach_menu_shm(&ex_menu_set);
1164
1165 return 0;
1166 }
1167
1168 int section_aid_locations_save(int uid)
1169 {
1170 char filename[FILE_PATH_LEN];
1171 FILE *fp;
1172 int i;
1173 int ret = 0;
1174
1175 snprintf(filename, sizeof(filename), "%s/%d", VAR_SECTION_AID_LOC_DIR, uid);
1176
1177 if ((fp = fopen(filename, "wb")) == NULL)
1178 {
1179 log_error("fopen(%s, wb) error: %d\n", filename, errno);
1180 return -1;
1181 }
1182
1183 for (i = 0; i < p_section_list_pool->section_count; i++)
1184 {
1185 if (fwrite(&(p_section_list_pool->sections[i].sid), sizeof(p_section_list_pool->sections[i].sid), 1, fp) != 1)
1186 {
1187 log_error("fwrite(%s, sid) error\n", filename);
1188 ret = -2;
1189 break;
1190 }
1191
1192 if (fwrite(&(section_aid_locations[i]), sizeof(section_aid_locations[i]), 1, fp) != 1)
1193 {
1194 log_error("fwrite(%s, aid) error\n", filename);
1195 ret = -2;
1196 break;
1197 }
1198 }
1199
1200 if (fclose(fp) < 0)
1201 {
1202 log_error("fclose(%s) error: %d\n", filename, errno);
1203 ret = -1;
1204 }
1205
1206 return ret;
1207 }
1208
1209 int section_aid_locations_load(int uid)
1210 {
1211 char filename[FILE_PATH_LEN];
1212 FILE *fp;
1213 int i;
1214 int32_t sid;
1215 int32_t aid;
1216 SECTION_LIST *p_section;
1217 int ret = 0;
1218
1219 snprintf(filename, sizeof(filename), "%s/%d", VAR_SECTION_AID_LOC_DIR, uid);
1220
1221 if ((fp = fopen(filename, "rb")) == NULL)
1222 {
1223 if (errno == ENOENT) // file not exist
1224 {
1225 return 0;
1226 }
1227 log_error("fopen(%s, rb) error: %d\n", filename, errno);
1228 return -1;
1229 }
1230
1231 while (!feof(fp))
1232 {
1233 if (fread(&sid, sizeof(sid), 1, fp) != 1)
1234 {
1235 if (ferror(fp) == 0)
1236 {
1237 break;
1238 }
1239 log_error("fread(%s, sid) error: %d\n", filename, ferror(fp));
1240 ret = -2;
1241 break;
1242 }
1243
1244 if (fread(&aid, sizeof(aid), 1, fp) != 1)
1245 {
1246 if (ferror(fp) == 0)
1247 {
1248 break;
1249 }
1250 log_error("fread(%s, aid) error: %d\n", filename, ferror(fp));
1251 ret = -2;
1252 break;
1253 }
1254
1255 p_section = section_list_find_by_sid(sid);
1256 if (p_section == NULL)
1257 {
1258 continue; // skip section no longer exist
1259 }
1260
1261 i = get_section_index(p_section);
1262 if (i < 0)
1263 {
1264 log_error("get_section_index(sid=%d) error\n", sid);
1265 ret = -3;
1266 break;
1267 }
1268 section_aid_locations[i] = aid;
1269 }
1270
1271 if (fclose(fp) < 0)
1272 {
1273 log_error("fclose(%s) error: %d\n", filename, errno);
1274 ret = -1;
1275 }
1276
1277 return ret;
1278 }

webmaster@leafok.com
ViewVC Help
Powered by ViewVC 1.3.0-beta1