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

Diff of /lbbs/src/menu.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

Revision 1.54 by sysadm, Tue May 20 12:18:24 2025 UTC Revision 1.68 by sysadm, Tue Jun 24 10:01:24 2025 UTC
# Line 16  Line 16 
16    
17  #include "bbs.h"  #include "bbs.h"
18  #include "bbs_cmd.h"  #include "bbs_cmd.h"
 #include "user_priv.h"  
19  #include "bbs_cmd.h"  #include "bbs_cmd.h"
20  #include "menu.h"  #include "common.h"
 #include "log.h"  
21  #include "io.h"  #include "io.h"
22    #include "log.h"
23    #include "menu.h"
24  #include "screen.h"  #include "screen.h"
25  #include "common.h"  #include "user_priv.h"
 #include <string.h>  
 #include <stdio.h>  
26  #include <ctype.h>  #include <ctype.h>
 #include <stdlib.h>  
27  #include <errno.h>  #include <errno.h>
28    #include <stdio.h>
29    #include <stdlib.h>
30    #include <string.h>
31  #include <unistd.h>  #include <unistd.h>
 #include <sys/shm.h>  
32  #include <sys/ipc.h>  #include <sys/ipc.h>
33    #include <sys/shm.h>
34    
35  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"
36  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"
# Line 78  int load_menu(MENU_SET *p_menu_set, cons Line 78  int load_menu(MENU_SET *p_menu_set, cons
78    
79          if ((fin = fopen(conf_file, "r")) == NULL)          if ((fin = fopen(conf_file, "r")) == NULL)
80          {          {
81                  log_error("Open %s failed", conf_file);                  log_error("Open %s failed\n", conf_file);
82                  return -2;                  return -2;
83          }          }
84    
# Line 119  int load_menu(MENU_SET *p_menu_set, cons Line 119  int load_menu(MENU_SET *p_menu_set, cons
119          p_menu_set->menu_screen_count = 0;          p_menu_set->menu_screen_count = 0;
120          p_menu_set->choose_step = 0;          p_menu_set->choose_step = 0;
121          p_menu_set->menu_id_path[0] = 0;          p_menu_set->menu_id_path[0] = 0;
122            p_menu_set->allow_exit = 0;
123    
124          while (fgets(buffer, sizeof(buffer), fin))          while (fgets(buffer, sizeof(buffer), fin))
125          {          {
# Line 161  int load_menu(MENU_SET *p_menu_set, cons Line 162  int load_menu(MENU_SET *p_menu_set, cons
162                                  p_menu->title.show = 0;                                  p_menu->title.show = 0;
163                                  p_menu->screen_show = 0;                                  p_menu->screen_show = 0;
164                                  p_menu->page_item_limit = 0;                                  p_menu->page_item_limit = 0;
165                                    p_menu->use_filter = 0;
166                                    p_menu->filter_handler = NULL;
167    
168                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
169                                  if (q == NULL)                                  if (q == NULL)
# Line 653  int load_menu(MENU_SET *p_menu_set, cons Line 656  int load_menu(MENU_SET *p_menu_set, cons
656                                                          return -1;                                                          return -1;
657                                                  }                                                  }
658                                          }                                          }
659                                            else if (strcmp(p, "use_filter") == 0)
660                                            {
661                                                    p_menu->use_filter = 1;
662    
663                                                    // Check syntax
664                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
665                                                    if (q != NULL)
666                                                    {
667                                                            log_error("Unknown extra content in menu config line %d\n", fin_line);
668                                                            return -1;
669                                                    }
670                                            }
671                                  }                                  }
672                          }                          }
673                          else // BEGIN of menu screen                          else // BEGIN of menu screen
# Line 772  int load_menu(MENU_SET *p_menu_set, cons Line 787  int load_menu(MENU_SET *p_menu_set, cons
787                          log_error("Undefined menu screen [%s]\n", p);                          log_error("Undefined menu screen [%s]\n", p);
788                          return -1;                          return -1;
789                  }                  }
790    
791                    // Set menu->filter_handler of each menu pointing to filter
792                    if (p_menu->use_filter == 1)
793                    {
794                            if ((p_menu->filter_handler = get_cmd_handler(p_menu->name)) == NULL)
795                            {
796                                    log_error("Undefined menu filter handler [%s]\n", p_menu->name);
797                                    return -1;
798                            }
799                    }
800          }          }
801    
802          for (menu_item_id = 0; menu_item_id < p_menu_set->menu_item_count; menu_item_id++)          for (menu_item_id = 0; menu_item_id < p_menu_set->menu_item_count; menu_item_id++)
# Line 807  int load_menu(MENU_SET *p_menu_set, cons Line 832  int load_menu(MENU_SET *p_menu_set, cons
832          return 0;          return 0;
833  }  }
834    
835  static int display_menu_cursor(MENU_SET *p_menu_set, int show)  int display_menu_cursor(MENU_SET *p_menu_set, int show)
836  {  {
837          MENU_ID menu_id;          MENU_ID menu_id;
838          MENU_ITEM_ID menu_item_id;          MENU_ITEM_ID menu_item_id;
# Line 864  static int display_menu_current_page(MEN Line 889  static int display_menu_current_page(MEN
889          {          {
890                  if (p_menu->title.row == 0 && p_menu->title.col == 0)                  if (p_menu->title.row == 0 && p_menu->title.col == 0)
891                  {                  {
892                          show_top(p_menu->title.text);                          show_top(p_menu->title.text, BBS_name, "");
893                  }                  }
894                  else                  else
895                  {                  {
# Line 956  int display_menu(MENU_SET *p_menu_set) Line 981  int display_menu(MENU_SET *p_menu_set)
981                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
982                          return REDRAW;                          return REDRAW;
983                  }                  }
984                  return EXITBBS;                  return EXITMENU;
985          }          }
986    
987          menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];          menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];
# Line 965  int display_menu(MENU_SET *p_menu_set) Line 990  int display_menu(MENU_SET *p_menu_set)
990          if (p_menu_item == NULL)          if (p_menu_item == NULL)
991          {          {
992                  log_error("get_menu_item_by_id(%d) return NULL pointer\n", menu_item_id);                  log_error("get_menu_item_by_id(%d) return NULL pointer\n", menu_item_id);
993                  menu_item_pos = 0;                  return EXITMENU;
994          }          }
995    
996          if (menu_item_pos > 0 &&          if (menu_item_pos > 0 &&
997                  checkpriv(&BBS_priv, 0, p_menu_item->priv) != 0 &&                  !(p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
998                  checklevel2(&BBS_priv, p_menu_item->level))                                                           : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
999                                                                    checklevel2(&BBS_priv, p_menu_item->level) == 0)))
1000          {          {
1001                  menu_selectable = 1;                  menu_selectable = 1;
1002          }          }
# Line 991  int display_menu(MENU_SET *p_menu_set) Line 1017  int display_menu(MENU_SET *p_menu_set)
1017    
1018                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;
1019    
1020                  if (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 || checklevel2(&BBS_priv, p_menu_item->level) == 0)                  if (p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
1021                                                               : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1022                                                                      checklevel2(&BBS_priv, p_menu_item->level) == 0))
1023                  {                  {
1024                          p_menu_set->menu_item_display[menu_item_pos] = 0;                          p_menu_set->menu_item_display[menu_item_pos] = 0;
1025                          p_menu_set->menu_item_r_row[menu_item_pos] = 0;                          p_menu_set->menu_item_r_row[menu_item_pos] = 0;
# Line 1025  int display_menu(MENU_SET *p_menu_set) Line 1053  int display_menu(MENU_SET *p_menu_set)
1053    
1054          if (!menu_selectable)          if (!menu_selectable)
1055          {          {
1056                  log_error("No selectable menu item in current menu (%s)\n", p_menu->name);                  moveto(p_menu->screen_row, p_menu->screen_col);
1057                    clrtoeol();
1058                    prints("没有可选项");
1059                    press_any_key();
1060                  return -1;                  return -1;
1061          }          }
1062    
# Line 1070  int menu_control(MENU_SET *p_menu_set, i Line 1101  int menu_control(MENU_SET *p_menu_set, i
1101    
1102          if (p_menu->item_count == 0)          if (p_menu->item_count == 0)
1103          {          {
1104    #ifdef _DEBUG
1105                  log_error("Empty menu (%s)\n", p_menu->name);                  log_error("Empty menu (%s)\n", p_menu->name);
1106    #endif
1107                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1108                  {                  {
1109                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
# Line 1116  int menu_control(MENU_SET *p_menu_set, i Line 1149  int menu_control(MENU_SET *p_menu_set, i
1149                          return ((*(p_menu_item->action_cmd_handler))((void *)(p_menu_item->name)));                          return ((*(p_menu_item->action_cmd_handler))((void *)(p_menu_item->name)));
1150                  }                  }
1151                  break;                  break;
1152            case KEY_ESC:
1153          case KEY_LEFT:          case KEY_LEFT:
1154                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1155                  {                  {
# Line 1131  int menu_control(MENU_SET *p_menu_set, i Line 1165  int menu_control(MENU_SET *p_menu_set, i
1165                  }                  }
1166                  else                  else
1167                  {                  {
1168                            if (p_menu_set->allow_exit)
1169                            {
1170                                    return EXITMENU;
1171                            }
1172    
1173                          display_menu_cursor(p_menu_set, 0);                          display_menu_cursor(p_menu_set, 0);
1174                          menu_item_pos = p_menu->item_count - 1;                          menu_item_pos = p_menu->item_count - 1;
1175                          while (menu_item_pos >= 0)                          while (menu_item_pos >= 0)
# Line 1306  int menu_control(MENU_SET *p_menu_set, i Line 1345  int menu_control(MENU_SET *p_menu_set, i
1345    
1346  int unload_menu(MENU_SET *p_menu_set)  int unload_menu(MENU_SET *p_menu_set)
1347  {  {
1348            int shmid;
1349    
1350            if (p_menu_set == NULL)
1351            {
1352                    return -1;
1353            }
1354    
1355          if (p_menu_set->p_menu_name_dict != NULL)          if (p_menu_set->p_menu_name_dict != NULL)
1356          {          {
1357                  trie_dict_destroy(p_menu_set->p_menu_name_dict);                  trie_dict_destroy(p_menu_set->p_menu_name_dict);
# Line 1318  int unload_menu(MENU_SET *p_menu_set) Line 1364  int unload_menu(MENU_SET *p_menu_set)
1364                  p_menu_set->p_menu_screen_dict = NULL;                  p_menu_set->p_menu_screen_dict = NULL;
1365          }          }
1366    
1367          unload_menu_shm(p_menu_set);          shmid = p_menu_set->shmid;
1368    
1369            detach_menu_shm(p_menu_set);
1370    
1371          if (shmctl(p_menu_set->shmid, IPC_RMID, NULL) == -1)          if (shmctl(shmid, IPC_RMID, NULL) == -1)
1372          {          {
1373                  log_error("shmctl(shmid=%d, IPC_RMID) error (%d)\n", p_menu_set->shmid, errno);                  log_error("shmctl(shmid=%d, IPC_RMID) error (%d)\n", shmid, errno);
1374                  return -1;                  return -1;
1375          }          }
1376    
1377          return 0;          return 0;
1378  }  }
1379    
1380  int load_menu_shm(MENU_SET *p_menu_set)  int get_menu_shm_readonly(MENU_SET *p_menu_set)
1381  {  {
1382          void *p_shm;          void *p_shm;
1383    
         // Mount shared memory  
1384          p_shm = shmat(p_menu_set->shmid, NULL, SHM_RDONLY);          p_shm = shmat(p_menu_set->shmid, NULL, SHM_RDONLY);
1385          if (p_shm == (void *)-1)          if (p_shm == (void *)-1)
1386          {          {
1387                  log_error("shmat() error (%d)\n", errno);                  log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);
1388                  return -1;                  return -1;
1389          }          }
1390    
         if (p_menu_set->p_reserved != NULL && shmdt(p_menu_set->p_reserved) == -1)  
         {  
                 log_error("shmdt() error (%d)\n", errno);  
                 return -2;  
         }  
1391          p_menu_set->p_reserved = p_shm;          p_menu_set->p_reserved = p_shm;
   
1392          p_menu_set->p_menu_pool = p_menu_set->p_reserved + MENU_SET_RESERVED_LENGTH;          p_menu_set->p_menu_pool = p_menu_set->p_reserved + MENU_SET_RESERVED_LENGTH;
1393          p_menu_set->p_menu_item_pool = p_menu_set->p_menu_pool + sizeof(MENU) * MAX_MENUS;          p_menu_set->p_menu_item_pool = p_menu_set->p_menu_pool + sizeof(MENU) * MAX_MENUS;
1394          p_menu_set->p_menu_screen_pool = p_menu_set->p_menu_item_pool + sizeof(MENU_ITEM) * MAX_MENUITEMS;          p_menu_set->p_menu_screen_pool = p_menu_set->p_menu_item_pool + sizeof(MENU_ITEM) * MAX_MENUITEMS;
1395          p_menu_set->p_menu_screen_buf = p_menu_set->p_menu_screen_pool + sizeof(MENU_SCREEN) * MAX_MENUS;          p_menu_set->p_menu_screen_buf = p_menu_set->p_menu_screen_pool + sizeof(MENU_SCREEN) * MAX_MENUS;
1396          p_menu_set->p_menu_screen_buf_free = p_menu_set->p_menu_screen_buf;          p_menu_set->p_menu_screen_buf_free = p_menu_set->p_menu_screen_buf;
1397    
1398          // Restore status varaibles into reserved memory area          return 0;
1399          p_menu_set->menu_count = *((int16_t *)p_menu_set->p_reserved);  }
1400          p_menu_set->menu_item_count = *(((int16_t *)p_menu_set->p_reserved) + 1);  
1401          p_menu_set->menu_screen_count = *(((int16_t *)p_menu_set->p_reserved) + 2);  int set_menu_shm_readonly(MENU_SET *p_menu_set)
1402    {
1403            void *p_shm;
1404    
1405          p_menu_set->choose_step = 0;          // Remap shared memory in read-only mode
1406          p_menu_set->menu_id_path[0] = 0;          p_shm = shmat(p_menu_set->shmid, p_menu_set->p_reserved, SHM_RDONLY | SHM_REMAP);
1407            if (p_shm == (void *)-1)
1408            {
1409                    log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);
1410                    return -1;
1411            }
1412    
1413          p_menu_set->p_menu_name_dict = NULL;          p_menu_set->p_reserved = p_shm;
         p_menu_set->p_menu_screen_dict = NULL;  
1414    
1415          return 0;          return 0;
1416  }  }
1417    
1418  int unload_menu_shm(MENU_SET *p_menu_set)  int detach_menu_shm(MENU_SET *p_menu_set)
1419  {  {
1420          p_menu_set->menu_count = 0;          p_menu_set->menu_count = 0;
1421          p_menu_set->menu_item_count = 0;          p_menu_set->menu_item_count = 0;
# Line 1381  int unload_menu_shm(MENU_SET *p_menu_set Line 1428  int unload_menu_shm(MENU_SET *p_menu_set
1428          p_menu_set->p_menu_screen_buf = NULL;          p_menu_set->p_menu_screen_buf = NULL;
1429          p_menu_set->p_menu_screen_buf_free = NULL;          p_menu_set->p_menu_screen_buf_free = NULL;
1430    
1431            p_menu_set->p_menu_name_dict = NULL;
1432            p_menu_set->p_menu_screen_dict = NULL;
1433    
1434          if (p_menu_set->p_reserved != NULL && shmdt(p_menu_set->p_reserved) == -1)          if (p_menu_set->p_reserved != NULL && shmdt(p_menu_set->p_reserved) == -1)
1435          {          {
1436                  log_error("shmdt() error (%d)\n", errno);                  log_error("shmdt() error (%d)\n", errno);
1437                  return -1;                  return -1;
1438          }          }
1439    
1440          p_menu_set->p_reserved = NULL;          p_menu_set->p_reserved = NULL;
1441    
1442          return 0;          return 0;


Legend:
Removed lines/characters  
Changed lines/characters
  Added lines/characters

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