/[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.49 by sysadm, Mon May 19 02:16:24 2025 UTC Revision 1.80 by sysadm, Tue Nov 4 13:49:51 2025 UTC
# Line 1  Line 1 
1  /***************************************************************************  /* SPDX-License-Identifier: GPL-3.0-or-later */
2                                                    menu.c  -  description  /*
3                                                           -------------------   * menu
4          Copyright            : (C) 2004-2025 by Leaflet   *   - configurable user interactive menu feature
5          Email                : leaflet@leafok.com   *
6   ***************************************************************************/   * Copyright (C) 2004-2025 by Leaflet <leaflet@leafok.com>
7     */
 /***************************************************************************  
  *                                                                         *  
  *   This program is free software; you can redistribute it and/or modify  *  
  *   it under the terms of the GNU General Public License as published by  *  
  *   the Free Software Foundation; either version 3 of the License, or     *  
  *   (at your option) any later version.                                   *  
  *                                                                         *  
  ***************************************************************************/  
8    
9  #include "bbs.h"  #include "bbs.h"
10  #include "bbs_cmd.h"  #include "bbs_cmd.h"
 #include "user_priv.h"  
11  #include "bbs_cmd.h"  #include "bbs_cmd.h"
12  #include "menu.h"  #include "common.h"
 #include "log.h"  
13  #include "io.h"  #include "io.h"
14    #include "log.h"
15    #include "menu.h"
16  #include "screen.h"  #include "screen.h"
17  #include "common.h"  #include "user_priv.h"
 #include <string.h>  
 #include <stdio.h>  
18  #include <ctype.h>  #include <ctype.h>
 #include <stdlib.h>  
19  #include <errno.h>  #include <errno.h>
20    #include <stdio.h>
21    #include <stdlib.h>
22    #include <string.h>
23  #include <unistd.h>  #include <unistd.h>
 #include <sys/shm.h>  
24  #include <sys/ipc.h>  #include <sys/ipc.h>
25    #include <sys/shm.h>
26    
27  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"
28  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"
# Line 38  Line 30 
30    
31  #define MENU_SET_RESERVED_LENGTH (sizeof(int16_t) * 4)  #define MENU_SET_RESERVED_LENGTH (sizeof(int16_t) * 4)
32    
33  MENU_SET *p_bbs_menu;  #define MENU_SHMGET_RETRY_LIMIT 10
34    
35    MENU_SET bbs_menu;
36    MENU_SET top10_menu;
37    
38  int load_menu(MENU_SET *p_menu_set, const char *conf_file)  int load_menu(MENU_SET *p_menu_set, const char *conf_file)
39  {  {
# Line 59  int load_menu(MENU_SET *p_menu_set, cons Line 54  int load_menu(MENU_SET *p_menu_set, cons
54          int proj_id;          int proj_id;
55          key_t key;          key_t key;
56          size_t size;          size_t size;
57            int retry_cnt;
58    
59            // Initialize the data structure
60            memset(p_menu_set, 0, sizeof(*p_menu_set));
61    
62          // Use trie_dict to search menu_id by menu name          // Use trie_dict to search menu_id by menu name
63          p_menu_set->p_menu_name_dict = trie_dict_create();          p_menu_set->p_menu_name_dict = trie_dict_create();
# Line 78  int load_menu(MENU_SET *p_menu_set, cons Line 77  int load_menu(MENU_SET *p_menu_set, cons
77    
78          if ((fin = fopen(conf_file, "r")) == NULL)          if ((fin = fopen(conf_file, "r")) == NULL)
79          {          {
80                  log_error("Open %s failed", conf_file);                  log_error("Open %s failed\n", conf_file);
81                  return -2;                  return -2;
82          }          }
83    
84          // Allocate shared memory          // Allocate shared memory
         proj_id = (int)(time(NULL) % getpid());  
         key = ftok(conf_file, proj_id);  
         if (key == -1)  
         {  
                 log_error("ftok(%s %d) error (%d)\n", conf_file, proj_id, errno);  
                 return -2;  
         }  
   
85          size = MENU_SET_RESERVED_LENGTH +          size = MENU_SET_RESERVED_LENGTH +
86                     sizeof(MENU) * MAX_MENUS +                     sizeof(MENU) * MAX_MENUS +
87                     sizeof(MENU_ITEM) * MAX_MENUITEMS +                     sizeof(MENU_ITEM) * MAX_MENUITEMS +
88                     sizeof(MENU_SCREEN) * MAX_MENUS +                     sizeof(MENU_SCREEN) * MAX_MENUS +
89                     MAX_MENU_SCR_BUF_LENGTH * MAX_MENUS;                     MAX_MENU_SCR_BUF_LENGTH * MAX_MENUS;
90          p_menu_set->shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0600);  
91          if (p_menu_set->shmid == -1)          proj_id = (int)(time(NULL) % getpid());
92            retry_cnt = 0;
93    
94            do
95          {          {
96                  log_error("shmget(size = %d) error (%d)\n", size, errno);                  key = ftok(conf_file, proj_id + retry_cnt);
97                  return -3;                  if (key == -1)
98          }                  {
99                            log_error("ftok(%s %d) error (%d)\n", conf_file, proj_id, errno);
100                            return -2;
101                    }
102    
103                    p_menu_set->shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0600);
104    
105                    if (p_menu_set->shmid == -1)
106                    {
107                            if (errno != EEXIST || retry_cnt + 1 >= MENU_SHMGET_RETRY_LIMIT)
108                            {
109                                    log_error("shmget(conf_file=%s, size=%d) error (%d) %d times\n",
110                                                      conf_file, size, errno, retry_cnt + 1);
111                                    break;
112                            }
113                            log_error("shmget(conf_file=%s, proj_id=%d, key=0x%x, size=%d) error (%d), retry ...\n",
114                                              conf_file, proj_id + retry_cnt, key, size, errno);
115                            retry_cnt++;
116                    }
117            } while (p_menu_set->shmid == -1);
118    
119          p_menu_set->p_reserved = shmat(p_menu_set->shmid, NULL, 0);          p_menu_set->p_reserved = shmat(p_menu_set->shmid, NULL, 0);
120          if (p_menu_set->p_reserved == (void *)-1)          if (p_menu_set->p_reserved == (void *)-1)
121          {          {
122                  log_error("shmat() error (%d)\n", errno);                  log_error("shmat() error (%d)\n", errno);
123                  return -3;                  return -3;
124          }          }
125          p_menu_set->p_menu_pool = p_menu_set->p_reserved + MENU_SET_RESERVED_LENGTH;  
126          p_menu_set->p_menu_item_pool = p_menu_set->p_menu_pool + sizeof(MENU) * MAX_MENUS;          p_menu_set->p_menu_pool = (char *)(p_menu_set->p_reserved) + MENU_SET_RESERVED_LENGTH;
127          p_menu_set->p_menu_screen_pool = p_menu_set->p_menu_item_pool + sizeof(MENU_ITEM) * MAX_MENUITEMS;          p_menu_set->p_menu_item_pool = (char *)(p_menu_set->p_menu_pool) + sizeof(MENU) * MAX_MENUS;
128          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_pool = (char *)(p_menu_set->p_menu_item_pool) + sizeof(MENU_ITEM) * MAX_MENUITEMS;
129            p_menu_set->p_menu_screen_buf = (char *)(p_menu_set->p_menu_screen_pool) + sizeof(MENU_SCREEN) * MAX_MENUS;
130          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;
131    
132          p_menu_set->menu_count = 0;          p_menu_set->menu_count = 0;
# Line 119  int load_menu(MENU_SET *p_menu_set, cons Line 134  int load_menu(MENU_SET *p_menu_set, cons
134          p_menu_set->menu_screen_count = 0;          p_menu_set->menu_screen_count = 0;
135          p_menu_set->choose_step = 0;          p_menu_set->choose_step = 0;
136          p_menu_set->menu_id_path[0] = 0;          p_menu_set->menu_id_path[0] = 0;
137            p_menu_set->menu_item_pos[0] = 0;
138            p_menu_set->allow_exit = 0;
139    
140          while (fgets(buffer, sizeof(buffer), fin))          while (fgets(buffer, sizeof(buffer), fin))
141          {          {
# Line 160  int load_menu(MENU_SET *p_menu_set, cons Line 177  int load_menu(MENU_SET *p_menu_set, cons
177                                  p_menu->item_count = 0;                                  p_menu->item_count = 0;
178                                  p_menu->title.show = 0;                                  p_menu->title.show = 0;
179                                  p_menu->screen_show = 0;                                  p_menu->screen_show = 0;
180                                    p_menu->page_item_limit = 0;
181                                    p_menu->use_filter = 0;
182                                    p_menu->filter_handler = NULL;
183    
184                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
185                                  if (q == NULL)                                  if (q == NULL)
# Line 585  int load_menu(MENU_SET *p_menu_set, cons Line 605  int load_menu(MENU_SET *p_menu_set, cons
605                                                          return -1;                                                          return -1;
606                                                  }                                                  }
607                                          }                                          }
608                                            else if (strcmp(p, "page") == 0)
609                                            {
610                                                    // Menu page row
611                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
612                                                    if (q == NULL)
613                                                    {
614                                                            log_error("Error menu page row in menu config line %d\n", fin_line);
615                                                            return -1;
616                                                    }
617                                                    p = q;
618                                                    while (isdigit(*q))
619                                                    {
620                                                            q++;
621                                                    }
622                                                    if (*q != '\0')
623                                                    {
624                                                            log_error("Error menu page row in menu config line %d\n", fin_line);
625                                                            return -1;
626                                                    }
627                                                    p_menu->page_row = (int16_t)atoi(p);
628    
629                                                    // Menu page col
630                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
631                                                    if (q == NULL)
632                                                    {
633                                                            log_error("Error menu page col in menu config line %d\n", fin_line);
634                                                            return -1;
635                                                    }
636                                                    p = q;
637                                                    while (isdigit(*q))
638                                                    {
639                                                            q++;
640                                                    }
641                                                    if (*q != '\0')
642                                                    {
643                                                            log_error("Error menu page col in menu config line %d\n", fin_line);
644                                                            return -1;
645                                                    }
646                                                    p_menu->page_col = (int16_t)atoi(p);
647    
648                                                    // Menu page item limit
649                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
650                                                    if (q == NULL)
651                                                    {
652                                                            log_error("Error menu page item limit in menu config line %d\n", fin_line);
653                                                            return -1;
654                                                    }
655                                                    p = q;
656                                                    while (isdigit(*q))
657                                                    {
658                                                            q++;
659                                                    }
660                                                    if (*q != '\0')
661                                                    {
662                                                            log_error("Error menu page item limit in menu config line %d\n", fin_line);
663                                                            return -1;
664                                                    }
665                                                    p_menu->page_item_limit = (int16_t)atoi(p);
666    
667                                                    // Check syntax
668                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
669                                                    if (q != NULL)
670                                                    {
671                                                            log_error("Unknown extra content in menu config line %d\n", fin_line);
672                                                            return -1;
673                                                    }
674                                            }
675                                            else if (strcmp(p, "use_filter") == 0)
676                                            {
677                                                    p_menu->use_filter = 1;
678    
679                                                    // Check syntax
680                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
681                                                    if (q != NULL)
682                                                    {
683                                                            log_error("Unknown extra content in menu config line %d\n", fin_line);
684                                                            return -1;
685                                                    }
686                                            }
687                                  }                                  }
688                          }                          }
689                          else // BEGIN of menu screen                          else // BEGIN of menu screen
# Line 704  int load_menu(MENU_SET *p_menu_set, cons Line 803  int load_menu(MENU_SET *p_menu_set, cons
803                          log_error("Undefined menu screen [%s]\n", p);                          log_error("Undefined menu screen [%s]\n", p);
804                          return -1;                          return -1;
805                  }                  }
806    
807                    // Set menu->filter_handler of each menu pointing to filter
808                    if (p_menu->use_filter == 1)
809                    {
810                            if ((p_menu->filter_handler = get_cmd_handler(p_menu->name)) == NULL)
811                            {
812                                    log_error("Undefined menu filter handler [%s]\n", p_menu->name);
813                                    return -1;
814                            }
815                    }
816          }          }
817    
818          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 739  int load_menu(MENU_SET *p_menu_set, cons Line 848  int load_menu(MENU_SET *p_menu_set, cons
848          return 0;          return 0;
849  }  }
850    
851  static int display_menu_cursor(MENU_SET *p_menu_set, int show)  int display_menu_cursor(MENU_SET *p_menu_set, int show)
852  {  {
853          MENU_ID menu_id;          MENU_ID menu_id;
854          MENU_ITEM_ID menu_item_id;          MENU_ITEM_ID menu_item_id;
# Line 770  static int display_menu_cursor(MENU_SET Line 879  static int display_menu_cursor(MENU_SET
879          return 0;          return 0;
880  }  }
881    
882  int display_menu(MENU_SET *p_menu_set)  static int display_menu_current_page(MENU_SET *p_menu_set)
883  {  {
884          int16_t row = 0;          int16_t row = 0;
885          int16_t col = 0;          int16_t col = 0;
         int menu_selectable = 0;  
886          MENU_ID menu_id;          MENU_ID menu_id;
887          MENU_ITEM_ID menu_item_id;          MENU_ITEM_ID menu_item_id;
888          MENU *p_menu;          MENU *p_menu;
889          MENU_ITEM *p_menu_item;          MENU_ITEM *p_menu_item;
         MENU_SCREEN *p_menu_screen;  
890          int16_t menu_item_pos;          int16_t menu_item_pos;
891            int16_t page_id = 0;
892            MENU_SCREEN *p_menu_screen;
893    
894          menu_id = p_menu_set->menu_id_path[p_menu_set->choose_step];          menu_id = p_menu_set->menu_id_path[p_menu_set->choose_step];
895          p_menu = get_menu_by_id(p_menu_set, menu_id);          p_menu = get_menu_by_id(p_menu_set, menu_id);
# Line 790  int display_menu(MENU_SET *p_menu_set) Line 899  int display_menu(MENU_SET *p_menu_set)
899                  return -1;                  return -1;
900          }          }
901    
902          menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];          clrline(p_menu->page_row, p_menu->page_row + p_menu->page_item_limit - 1);
         menu_item_id = p_menu->items[menu_item_pos];  
         p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);  
         if (p_menu_item == NULL)  
         {  
                 log_error("get_menu_item_by_id(%d) return NULL pointer\n", menu_item_id);  
                 return -1;  
         }  
   
         if (menu_item_pos > 0 &&  
                 checkpriv(&BBS_priv, 0, p_menu_item->priv) != 0 &&  
                 checklevel2(&BBS_priv, p_menu_item->level))  
         {  
                 menu_selectable = 1;  
         }  
903    
904          if (p_menu->title.show)          if (p_menu->title.show)
905          {          {
906                  if (p_menu->title.row == 0 && p_menu->title.col == 0)                  if (p_menu->title.row == 0 && p_menu->title.col == 0)
907                  {                  {
908                          show_top(p_menu->title.text);                          show_top(p_menu->title.text, BBS_name, "");
909                  }                  }
910                  else                  else
911                  {                  {
# Line 835  int display_menu(MENU_SET *p_menu_set) Line 930  int display_menu(MENU_SET *p_menu_set)
930                  prints("%s", p_menu_set->p_menu_screen_buf + p_menu_screen->buf_offset);                  prints("%s", p_menu_set->p_menu_screen_buf + p_menu_screen->buf_offset);
931          }          }
932    
933            menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];
934            page_id = p_menu_set->menu_item_page_id[menu_item_pos];
935    
936            while (menu_item_pos >= 0)
937            {
938                    if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
939                    {
940                            menu_item_pos++;
941                            break;
942                    }
943    
944                    if (menu_item_pos == 0)
945                    {
946                            break;
947                    }
948    
949                    menu_item_pos--;
950            }
951    
952            for (; menu_item_pos < p_menu->item_count; menu_item_pos++)
953            {
954                    if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
955                    {
956                            break;
957                    }
958    
959                    menu_item_id = p_menu->items[menu_item_pos];
960                    p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);
961    
962                    if (p_menu_set->menu_item_display[menu_item_pos] == 0)
963                    {
964                            continue;
965                    }
966    
967                    row = p_menu_set->menu_item_r_row[menu_item_pos];
968                    col = p_menu_set->menu_item_r_col[menu_item_pos];
969    
970                    moveto(row, col);
971                    prints("%s", p_menu_item->text);
972            }
973    
974            return 0;
975    }
976    
977    int display_menu(MENU_SET *p_menu_set)
978    {
979            int16_t row = 0;
980            int16_t col = 0;
981            int menu_selectable = 0;
982            MENU_ID menu_id;
983            MENU_ITEM_ID menu_item_id;
984            MENU *p_menu;
985            MENU_ITEM *p_menu_item;
986            int16_t menu_item_pos;
987            int16_t page_id = 0;
988            int page_item_count = 0;
989    
990            menu_id = p_menu_set->menu_id_path[p_menu_set->choose_step];
991            p_menu = get_menu_by_id(p_menu_set, menu_id);
992            if (p_menu == NULL)
993            {
994                    log_error("get_menu_by_id(%d) return NULL pointer\n", menu_id);
995                    if (p_menu_set->choose_step > 0)
996                    {
997                            p_menu_set->choose_step--;
998                            return REDRAW;
999                    }
1000                    return EXITMENU;
1001            }
1002    
1003            if (p_menu->item_count <= 0) // empty menu
1004            {
1005                    moveto(p_menu->screen_row, p_menu->screen_col);
1006                    clrtoeol();
1007                    prints("没有可选项");
1008                    press_any_key();
1009                    return -1;
1010            }
1011    
1012            menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];
1013            menu_item_id = p_menu->items[menu_item_pos];
1014            p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);
1015            if (p_menu_item == NULL)
1016            {
1017                    log_error("get_menu_item_by_id(%d) return NULL pointer\n", menu_item_id);
1018                    return EXITMENU;
1019            }
1020    
1021            if (menu_item_pos > 0 &&
1022                    !(p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
1023                                                             : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1024                                                                    checklevel2(&BBS_priv, p_menu_item->level) == 0)))
1025            {
1026                    menu_selectable = 1;
1027            }
1028    
1029          for (menu_item_pos = 0; menu_item_pos < p_menu->item_count; menu_item_pos++)          for (menu_item_pos = 0; menu_item_pos < p_menu->item_count; menu_item_pos++)
1030          {          {
1031                  menu_item_id = p_menu->items[menu_item_pos];                  menu_item_id = p_menu->items[menu_item_pos];
# Line 849  int display_menu(MENU_SET *p_menu_set) Line 1040  int display_menu(MENU_SET *p_menu_set)
1040                          col = p_menu_item->col;                          col = p_menu_item->col;
1041                  }                  }
1042    
1043                  if (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 || checklevel(&BBS_priv, p_menu_item->level) == 0)                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;
1044    
1045                    if (p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
1046                                                               : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1047                                                                      checklevel2(&BBS_priv, p_menu_item->level) == 0))
1048                  {                  {
1049                          p_menu_set->menu_item_display[menu_item_pos] = 0;                          p_menu_set->menu_item_display[menu_item_pos] = 0;
1050                          p_menu_set->menu_item_r_row[menu_item_pos] = 0;                          p_menu_set->menu_item_r_row[menu_item_pos] = 0;
# Line 868  int display_menu(MENU_SET *p_menu_set) Line 1063  int display_menu(MENU_SET *p_menu_set)
1063                          p_menu_set->menu_item_r_row[menu_item_pos] = row;                          p_menu_set->menu_item_r_row[menu_item_pos] = row;
1064                          p_menu_set->menu_item_r_col[menu_item_pos] = col;                          p_menu_set->menu_item_r_col[menu_item_pos] = col;
1065    
                         moveto(row, col);  
                         prints("%s", p_menu_item->text);  
   
1066                          row++;                          row++;
1067    
1068                            page_item_count++;
1069                            if (p_menu->page_item_limit > 0 && page_item_count >= p_menu->page_item_limit)
1070                            {
1071                                    page_id++;
1072                                    page_item_count = 0;
1073                                    row = p_menu->page_row;
1074                                    col = p_menu->page_col;
1075                            }
1076                  }                  }
1077          }          }
1078    
1079          if (!menu_selectable)          if (!menu_selectable)
1080          {          {
1081                    moveto(p_menu->screen_row, p_menu->screen_col);
1082                    clrtoeol();
1083                    prints("没有可选项");
1084                    press_any_key();
1085                    return -1;
1086            }
1087    
1088            if (display_menu_current_page(p_menu_set) != 0)
1089            {
1090                  return -1;                  return -1;
1091          }          }
1092    
# Line 892  int menu_control(MENU_SET *p_menu_set, i Line 1102  int menu_control(MENU_SET *p_menu_set, i
1102          MENU *p_menu;          MENU *p_menu;
1103          MENU_ITEM *p_menu_item;          MENU_ITEM *p_menu_item;
1104          int16_t menu_item_pos;          int16_t menu_item_pos;
1105            int16_t page_id;
1106            int require_page_change = 0;
1107    
1108          if (p_menu_set->menu_count == 0)          if (p_menu_set->menu_count == 0)
1109          {          {
1110                  return 0;                  log_error("Empty menu set\n");
1111                    return EXITBBS;
1112          }          }
1113    
1114          menu_id = p_menu_set->menu_id_path[p_menu_set->choose_step];          menu_id = p_menu_set->menu_id_path[p_menu_set->choose_step];
# Line 903  int menu_control(MENU_SET *p_menu_set, i Line 1116  int menu_control(MENU_SET *p_menu_set, i
1116          if (p_menu == NULL)          if (p_menu == NULL)
1117          {          {
1118                  log_error("get_menu_by_id(%d) return NULL pointer\n", menu_id);                  log_error("get_menu_by_id(%d) return NULL pointer\n", menu_id);
1119                  return -1;                  if (p_menu_set->choose_step > 0)
1120                    {
1121                            p_menu_set->choose_step--;
1122                            return REDRAW;
1123                    }
1124                    return EXITBBS;
1125          }          }
1126    
1127          if (p_menu->item_count == 0)          if (p_menu->item_count == 0)
1128          {          {
1129                  return 0;  #ifdef _DEBUG
1130                    log_error("Empty menu (%s)\n", p_menu->name);
1131    #endif
1132                    if (p_menu_set->choose_step > 0)
1133                    {
1134                            p_menu_set->choose_step--;
1135                            return REDRAW;
1136                    }
1137                    return EXITBBS;
1138          }          }
1139    
1140          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];
1141            page_id = p_menu_set->menu_item_page_id[menu_item_pos];
1142    
1143          menu_item_id = p_menu->items[menu_item_pos];          menu_item_id = p_menu->items[menu_item_pos];
1144          p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);          p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);
1145          if (p_menu_item == NULL)          if (p_menu_item == NULL)
1146          {          {
1147                  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);
1148                  return -1;                  p_menu_set->menu_item_pos[p_menu_set->choose_step] = 0;
1149                    return REDRAW;
1150          }          }
1151    
1152          switch (key)          switch (key)
1153          {          {
1154          case CR:          case CR:
                 igetch_reset();  
1155          case KEY_RIGHT:          case KEY_RIGHT:
1156                  if (p_menu_item->submenu)                  if (p_menu_item->submenu)
1157                  {                  {
# Line 945  int menu_control(MENU_SET *p_menu_set, i Line 1173  int menu_control(MENU_SET *p_menu_set, i
1173                          return ((*(p_menu_item->action_cmd_handler))((void *)(p_menu_item->name)));                          return ((*(p_menu_item->action_cmd_handler))((void *)(p_menu_item->name)));
1174                  }                  }
1175                  break;                  break;
1176            case KEY_ESC:
1177          case KEY_LEFT:          case KEY_LEFT:
1178                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1179                  {                  {
1180                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
1181                          if (display_menu(p_menu_set) != 0)                          return REDRAW;
                         {  
                                 return menu_control(p_menu_set, KEY_LEFT);  
                         }  
                         if (p_menu_set->choose_step == 0)  
                         {  
                                 return REDRAW;  
                         }  
1182                  }                  }
1183                  else                  else
1184                  {                  {
1185                            if (p_menu_set->allow_exit)
1186                            {
1187                                    return EXITMENU;
1188                            }
1189    
1190                          display_menu_cursor(p_menu_set, 0);                          display_menu_cursor(p_menu_set, 0);
1191                          menu_item_pos = p_menu->item_count - 1;                          menu_item_pos = p_menu->item_count - 1;
1192                          while (menu_item_pos >= 0)                          while (menu_item_pos >= 0)
# Line 985  int menu_control(MENU_SET *p_menu_set, i Line 1212  int menu_control(MENU_SET *p_menu_set, i
1212                          display_menu_cursor(p_menu_set, 1);                          display_menu_cursor(p_menu_set, 1);
1213                  }                  }
1214                  break;                  break;
1215            case KEY_PGUP:
1216                    require_page_change = 1;
1217          case KEY_UP:          case KEY_UP:
1218                  display_menu_cursor(p_menu_set, 0);                  display_menu_cursor(p_menu_set, 0);
1219                  do                  do
# Line 993  int menu_control(MENU_SET *p_menu_set, i Line 1222  int menu_control(MENU_SET *p_menu_set, i
1222                          if (menu_item_pos < 0)                          if (menu_item_pos < 0)
1223                          {                          {
1224                                  menu_item_pos = p_menu->item_count - 1;                                  menu_item_pos = p_menu->item_count - 1;
1225                                    require_page_change = 0;
1226                          }                          }
1227                          menu_item_id = p_menu->items[menu_item_pos];                          menu_item_id = p_menu->items[menu_item_pos];
1228                          p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);                          p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);
# Line 1001  int menu_control(MENU_SET *p_menu_set, i Line 1231  int menu_control(MENU_SET *p_menu_set, i
1231                                  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);
1232                                  return -1;                                  return -1;
1233                          }                          }
1234                  } while (!p_menu_set->menu_item_display[menu_item_pos]);                          if (require_page_change && p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1235                            {
1236                                    require_page_change = 0;
1237                            }
1238                    } while (require_page_change || !p_menu_set->menu_item_display[menu_item_pos]);
1239                  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] = menu_item_pos;
1240                    if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1241                    {
1242                            display_menu_current_page(p_menu_set);
1243                    }
1244                  display_menu_cursor(p_menu_set, 1);                  display_menu_cursor(p_menu_set, 1);
1245                  break;                  break;
1246            case KEY_PGDN:
1247                    require_page_change = 1;
1248          case KEY_DOWN:          case KEY_DOWN:
1249                  display_menu_cursor(p_menu_set, 0);                  display_menu_cursor(p_menu_set, 0);
1250                  do                  do
# Line 1013  int menu_control(MENU_SET *p_menu_set, i Line 1253  int menu_control(MENU_SET *p_menu_set, i
1253                          if (menu_item_pos >= p_menu->item_count)                          if (menu_item_pos >= p_menu->item_count)
1254                          {                          {
1255                                  menu_item_pos = 0;                                  menu_item_pos = 0;
1256                                    require_page_change = 0;
1257                          }                          }
1258                          menu_item_id = p_menu->items[menu_item_pos];                          menu_item_id = p_menu->items[menu_item_pos];
1259                          p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);                          p_menu_item = get_menu_item_by_id(p_menu_set, menu_item_id);
# Line 1021  int menu_control(MENU_SET *p_menu_set, i Line 1262  int menu_control(MENU_SET *p_menu_set, i
1262                                  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);
1263                                  return -1;                                  return -1;
1264                          }                          }
1265                  } while (!p_menu_set->menu_item_display[menu_item_pos]);                          if (require_page_change && p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1266                            {
1267                                    require_page_change = 0;
1268                            }
1269                    } while (require_page_change || !p_menu_set->menu_item_display[menu_item_pos]);
1270                  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] = menu_item_pos;
1271                    if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1272                    {
1273                            display_menu_current_page(p_menu_set);
1274                    }
1275                  display_menu_cursor(p_menu_set, 1);                  display_menu_cursor(p_menu_set, 1);
1276                  break;                  break;
1277          case KEY_HOME:          case KEY_HOME:
         case KEY_PGUP:  
1278                  display_menu_cursor(p_menu_set, 0);                  display_menu_cursor(p_menu_set, 0);
1279                  menu_item_pos = 0;                  menu_item_pos = 0;
1280                  while (menu_item_pos < p_menu->item_count - 1)                  while (menu_item_pos < p_menu->item_count - 1)
# Line 1047  int menu_control(MENU_SET *p_menu_set, i Line 1295  int menu_control(MENU_SET *p_menu_set, i
1295                          menu_item_pos++;                          menu_item_pos++;
1296                  }                  }
1297                  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] = menu_item_pos;
1298                    if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1299                    {
1300                            display_menu_current_page(p_menu_set);
1301                    }
1302                  display_menu_cursor(p_menu_set, 1);                  display_menu_cursor(p_menu_set, 1);
1303                  break;                  break;
1304          case KEY_END:          case KEY_END:
         case KEY_PGDN:  
1305                  display_menu_cursor(p_menu_set, 0);                  display_menu_cursor(p_menu_set, 0);
1306                  menu_item_pos = p_menu->item_count - 1;                  menu_item_pos = p_menu->item_count - 1;
1307                  while (menu_item_pos > 0)                  while (menu_item_pos > 0)
# Line 1071  int menu_control(MENU_SET *p_menu_set, i Line 1322  int menu_control(MENU_SET *p_menu_set, i
1322                          menu_item_pos--;                          menu_item_pos--;
1323                  }                  }
1324                  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] = menu_item_pos;
1325                    if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1326                    {
1327                            display_menu_current_page(p_menu_set);
1328                    }
1329                  display_menu_cursor(p_menu_set, 1);                  display_menu_cursor(p_menu_set, 1);
1330                  break;                  break;
1331          default:          default:
# Line 1090  int menu_control(MENU_SET *p_menu_set, i Line 1345  int menu_control(MENU_SET *p_menu_set, i
1345                                  {                                  {
1346                                          display_menu_cursor(p_menu_set, 0);                                          display_menu_cursor(p_menu_set, 0);
1347                                          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] = menu_item_pos;
1348                                            if (p_menu_set->menu_item_page_id[menu_item_pos] != page_id)
1349                                            {
1350                                                    display_menu_current_page(p_menu_set);
1351                                            }
1352                                          display_menu_cursor(p_menu_set, 1);                                          display_menu_cursor(p_menu_set, 1);
1353                                          return 0;                                          break;
1354                                  }                                  }
1355                          }                          }
1356                  }                  }
# Line 1103  int menu_control(MENU_SET *p_menu_set, i Line 1362  int menu_control(MENU_SET *p_menu_set, i
1362    
1363  int unload_menu(MENU_SET *p_menu_set)  int unload_menu(MENU_SET *p_menu_set)
1364  {  {
1365            int shmid;
1366    
1367            if (p_menu_set == NULL)
1368            {
1369                    return -1;
1370            }
1371    
1372          if (p_menu_set->p_menu_name_dict != NULL)          if (p_menu_set->p_menu_name_dict != NULL)
1373          {          {
1374                  trie_dict_destroy(p_menu_set->p_menu_name_dict);                  trie_dict_destroy(p_menu_set->p_menu_name_dict);
# Line 1115  int unload_menu(MENU_SET *p_menu_set) Line 1381  int unload_menu(MENU_SET *p_menu_set)
1381                  p_menu_set->p_menu_screen_dict = NULL;                  p_menu_set->p_menu_screen_dict = NULL;
1382          }          }
1383    
1384          unload_menu_shm(p_menu_set);          shmid = p_menu_set->shmid;
1385    
1386          if (shmctl(p_menu_set->shmid, IPC_RMID, NULL) == -1)          detach_menu_shm(p_menu_set);
1387    
1388            if (shmctl(shmid, IPC_RMID, NULL) == -1)
1389          {          {
1390                  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);
1391                  return -1;                  return -1;
1392          }          }
1393    
1394          return 0;          return 0;
1395  }  }
1396    
1397  int load_menu_shm(MENU_SET *p_menu_set)  int get_menu_shm_readonly(MENU_SET *p_menu_set)
1398  {  {
1399          // Mount shared memory          void *p_shm;
1400          if (p_menu_set->p_reserved == NULL)  
1401            p_shm = shmat(p_menu_set->shmid, NULL, SHM_RDONLY);
1402            if (p_shm == (void *)-1)
1403          {          {
1404                  p_menu_set->p_reserved = shmat(p_menu_set->shmid, NULL, SHM_RDONLY);                  log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);
1405                  if (p_menu_set->p_reserved == (void *)-1)                  return -1;
                 {  
                         log_error("shmat() error (%d)\n", errno);  
                         return -1;  
                 }  
1406          }          }
1407    
1408          p_menu_set->p_menu_pool = p_menu_set->p_reserved + MENU_SET_RESERVED_LENGTH;          p_menu_set->p_reserved = p_shm;
1409          p_menu_set->p_menu_item_pool = p_menu_set->p_menu_pool + sizeof(MENU) * MAX_MENUS;          p_menu_set->p_menu_pool = (char *)(p_menu_set->p_reserved) + MENU_SET_RESERVED_LENGTH;
1410          p_menu_set->p_menu_screen_pool = p_menu_set->p_menu_item_pool + sizeof(MENU_ITEM) * MAX_MENUITEMS;          p_menu_set->p_menu_item_pool = (char *)(p_menu_set->p_menu_pool) + sizeof(MENU) * MAX_MENUS;
1411          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_pool = (char *)(p_menu_set->p_menu_item_pool) + sizeof(MENU_ITEM) * MAX_MENUITEMS;
1412            p_menu_set->p_menu_screen_buf = (char *)(p_menu_set->p_menu_screen_pool) + sizeof(MENU_SCREEN) * MAX_MENUS;
1413          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;
1414    
         // Restore status varaibles into reserved memory area  
         p_menu_set->menu_count = *((int16_t *)p_menu_set->p_reserved);  
         p_menu_set->menu_item_count = *(((int16_t *)p_menu_set->p_reserved) + 1);  
         p_menu_set->menu_screen_count = *(((int16_t *)p_menu_set->p_reserved) + 2);  
   
1415          p_menu_set->choose_step = 0;          p_menu_set->choose_step = 0;
1416          p_menu_set->menu_id_path[0] = 0;          p_menu_set->menu_id_path[0] = 0;
1417            p_menu_set->menu_item_pos[0] = 0;
1418    
1419          p_menu_set->p_menu_name_dict = NULL;          return 0;
1420          p_menu_set->p_menu_screen_dict = NULL;  }
1421    
1422    int set_menu_shm_readonly(MENU_SET *p_menu_set)
1423    {
1424            void *p_shm;
1425    
1426            // Remap shared memory in read-only mode
1427            p_shm = shmat(p_menu_set->shmid, p_menu_set->p_reserved, SHM_RDONLY | SHM_REMAP);
1428            if (p_shm == (void *)-1)
1429            {
1430                    log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);
1431                    return -1;
1432            }
1433    
1434            p_menu_set->p_reserved = p_shm;
1435    
1436          return 0;          return 0;
1437  }  }
1438    
1439  int unload_menu_shm(MENU_SET *p_menu_set)  int detach_menu_shm(MENU_SET *p_menu_set)
1440  {  {
1441          p_menu_set->menu_count = 0;          p_menu_set->menu_count = 0;
1442          p_menu_set->menu_item_count = 0;          p_menu_set->menu_item_count = 0;
# Line 1172  int unload_menu_shm(MENU_SET *p_menu_set Line 1449  int unload_menu_shm(MENU_SET *p_menu_set
1449          p_menu_set->p_menu_screen_buf = NULL;          p_menu_set->p_menu_screen_buf = NULL;
1450          p_menu_set->p_menu_screen_buf_free = NULL;          p_menu_set->p_menu_screen_buf_free = NULL;
1451    
1452            p_menu_set->p_menu_name_dict = NULL;
1453            p_menu_set->p_menu_screen_dict = NULL;
1454    
1455          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)
1456          {          {
1457                  log_error("shmdt() error (%d)\n", errno);                  log_error("shmdt() error (%d)\n", errno);
1458                  return -1;                  return -1;
1459          }          }
1460    
1461          p_menu_set->p_reserved = NULL;          p_menu_set->p_reserved = NULL;
1462    
1463          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