summaryrefslogblamecommitdiffstats
path: root/mbbsd/announce.c
blob: 3bb1aad5d6c49d19cbd04c4f7c3ce493258cbfa3 (plain) (tree)
1
2
3
4
5
6
7
8
9
          

                

                       


                                     
 
    

                                                           

                             
              


                                 
               
                                                         




                                            
           




                                 
                           


                                                                    

 
           



                                         


                                                                                            
           






                                   




                                                        
                  

                               
                                                                 


                                  


                                                                         
                                                                          

                                     



                                                                 














                                                                             
          




                                        
                                                                                     

                     
                       
 
                                      
 



                       
                           
                    
                         
                              
                                                          


                                           
                                                                       
                       
                             


                   


                          
 
           

                     










                                                            
                           
                                                 
                                                                





                                                                 
                         






                                                 
           



                                                     

                                                                   
                                           











                             
           



                                               
                           
                                                 

                        

                                        

                                                                            

             


                                                             



                         
                         
 
           

                                
                        

                                                      
                                             

      


                                                                 
 
                                            
 
                   

                                
                                                                  
              
 

                               
                                                                  
              



                                                                    
                                 




                                       
                          

                             

                                  
                   
                                                                              
                                                      

                      
                                                            
                                                              

                      
                                                        
                                           

                      
                                                        
                                               

                      
                               
                                                                          

                                      
                                                                          

                      
                                                 

                      

                             






                                       

                                                                           
                         
            


                          
                   
                 
                                          






                          
                                          




                                           

     
                                                          


                         
           






                                     

                         



                                                                        




                                                                   
                   

                                                               
                                                                   
              

                            
                                                        
 

                                                               
 
                          


                                           
                                                         

                                                       
                                                                         
                                           
                                          

                                            
                                            
                 
                                         

                                           
                                        
                                         

                                           


                                                                         
                            




                                   


                                                                      








                                             
           





                                     

                         

                              
                                                             


                                                             

                                                                             
                                                                           
                 


                                                           


                                                             
                                      

                                                                        
                                                                  


                                                                  






















                                               
          


                                     
                              


                                              
 






                                          

                    

                   
 

                                           



                                                          












                                                      
     

               
 
 
           






                                  
 
                                                                          
                                                                      

                                                       
                       
                          
                     






                                                 
                                                   
                 

                                                                                   
                     
                                                                   

                     
                                                                   
                     
                                                                                   

                     
              




                                                         
           


                               
 
                                                        



                                           
           




                                                               
 

                                                                         
                           


                                                            
                                                                        
                                          
                          
                   
                                                        


                                                                             
                                          
                          
                   
                                                        

                      
                              

                                                                       


                                                        
                   


                                 



                                                                  
 

                                            
                    

                                                    
                              
                                                                         
                                     
                          
                   
                                                        

                   

                                 
 

                                                                
                    
 



                                                                  



                                                    
                                                                        
                                          
                          
                   
                                                        




                   
           



                                 

                                                         
                                              
                                                                   
                                                             



                                                         
           










                                                           


                                                    
           



                                 

                                                         
                                                                     
                                                              







                                                         
           





                                 

                         

                                                                         
                     

                                                        
                                                           
                            


                                                                 


                            


                                                                               

                                
                          

                                         
                                                                      


                 
                          
                                                                           
                        





                                                                           
     
                        
 
           

        




                                                                       
      
 
          
                          
 


                                                                




                                                                        

                 
 
   

                                                  
                                                     



                                                 
                        
 
              
                                                       
                   
                                                     

                                          
 
                                                 




                                            

                                      

                   

                             
                                
                       

                       
                                                              



                                                                            
                                                 

                                                     
                  


                                                    



                            
                     

                    
                             

                                    
 

                      
                                   




                           
                                  









                                    
                                          
                                  
                                         













                                                   
                                




                                 





                           

                 

                                                                          
                                                      
                                


                                                   
 
                                                        


                                            



                                                                          

                                                                       
                                         

                                                                          





                               


                                        
 

                                                               






                                                                        
                                                          
                                                           


                                                                               


                               



                       



                                                                    


                                                          
                                                                              
                                                                         

                                                                     
                                          
                                                
 
                                                              
                                                 

                                                                       


                                        


                                                                      
                                                             
                                                              
                                                

                                                                     
                                          
                                                        
                                                                     



                                                                  
                                 
                                                


                                                  

                                               


                                           

                                                      




                                                    

                                                

                                                                       
                                          

                                  

                                                                               
                                                
                                









                                          

                                                                          


                                                                                






                                         


                 
                                                                











                                                                          
                                                           






                                            

         

                                            
 
                         







                                         
                     
                                    







                                        
                                         














                                         


                             





                                    
                                           






                                    
                         








                                    

                                














                                        
                                           
 
   

          





                                                                       

           
                        





                                                                        

                                             



                                        
                                       
                                                   
                                                               
                   

                                                       
                   
                                     

                                                   


                                   

                                                        
                 
                                               







                                                        






                                                                   

                             
                      

                                                  






                                                                              



























                                                                           
            







                                                                 




















                                                                            
 




                     
                               




                             
/* $Id$ */
#include "bbs.h"

#define PATHLEN     256

static char     copyfile[PATHLEN];
static char     copytitle[TTLEN + 1];
static char     copyowner[IDLEN + 2];

void
a_copyitem(char *fpath, char *title, char *owner, int mode)
{
    strcpy(copyfile, fpath);
    strcpy(copytitle, title);
    if (owner)
    strcpy(copyowner, owner);
    else
    *copyowner = 0;
    if (mode) {
    vmsg("檔案標記完成。[注意] 拷貝後才能刪除原文!");
    }
}

#define FHSZ            sizeof(fileheader_t)

static void
a_loadname(menu_t * pm)
{
    char            buf[PATHLEN];
    int             len;

    setadir(buf, pm->path);
    len = get_records(buf, pm->header, FHSZ, pm->page + 1, p_lines);
    if (len < p_lines)
    bzero(&pm->header[len], FHSZ * (p_lines - len));
}

static void
a_timestamp(char *buf, time_t * time)
{
    struct tm      *pt = localtime(time);

    sprintf(buf, "%02d/%02d/%02d", pt->tm_mon + 1, pt->tm_mday, (pt->tm_year + 1900) % 100);
}

static void
a_showmenu(menu_t * pm)
{
    char           *title, *editor;
    int             n;
    fileheader_t   *item;
    char            buf[PATHLEN];
    time_t          dtime;

    showtitle("精華文章", pm->mtitle);
    prints("   \033[1;36m編號    標      題%56s\033[0m",
       "編    選      日    期");

    if (pm->num) {
    setadir(buf, pm->path);
    a_loadname(pm);
    for (n = 0; n < p_lines && pm->page + n < pm->num; n++) {
        item = &pm->header[n];
        title = item->title;
        editor = item->owner;
        /*
         * Ptt 把時間改為取檔案時間 dtime = atoi(&item->filename[2]);
         */
        snprintf(buf, sizeof(buf), "%s/%s", pm->path, item->filename);
        dtime = dasht(buf);
        a_timestamp(buf, &dtime);
        prints("\n%6d%c %-47.46s%-13s[%s]", pm->page + n + 1,
           (item->filemode & FILE_BM) ? 'X' :
           (item->filemode & FILE_HIDE) ? ')' : '.',
           title, editor,
           buf);
    }
    } else
    outs("\n  《精華區》尚在吸取天地間的日月精華中... :)");

    move(b_lines, 1);
    outs(pm->level ?
     "\033[34;46m 【板  主】 \033[31;47m  (h)\033[30m說明  "
     "\033[31m(q/←)\033[30m離開  \033[31m(n)\033[30m新增文章  "
     "\033[31m(g)\033[30m新增目錄  \033[31m(e)\033[30m編輯檔案  \033[m" :
     "\033[34;46m 【功\能鍵】 \033[31;47m  (h)\033[30m說明  "
     "\033[31m(q/←)\033[30m離開  \033[31m(k↑j↓)\033[30m移動游標  "
     "\033[31m(enter/→)\033[30m讀取資料  \033[m");
}

static int
a_searchtitle(menu_t * pm, int rev)
{
    static char     search_str[40] = "";
    int             pos;

    getdata(b_lines - 1, 1, "[搜尋]關鍵字:", search_str, sizeof(search_str), DOECHO);

    if (!*search_str)
    return pm->now;

    str_lower(search_str, search_str);

    rev = rev ? -1 : 1;
    pos = pm->now;
    do {
    pos += rev;
    if (pos == pm->num)
        pos = 0;
    else if (pos < 0)
        pos = pm->num - 1;
    if (pos < pm->page || pos >= pm->page + p_lines) {
        pm->page = pos - pos % p_lines;
        a_loadname(pm);
    }
    if (strstr_lower(pm->header[pos - pm->page].title, search_str))
        return pos;
    } while (pos != pm->now);
    return pm->now;
}

enum {
    NOBODY, MANAGER, SYSOP
};

static void
a_showhelp(int level)
{
    clear();
    outs("\033[36m【 " BBSNAME "公佈欄使用說明 】\033[m\n\n"
     "[←][q]         離開到上一層目錄\n"
     "[↑][k]         上一個選項\n"
     "[↓][j]         下一個選項\n"
     "[→][r][enter]  進入目錄/讀取文章\n"
     "[^B][PgUp]      上頁選單\n"
     "[^F][PgDn][Spc] 下頁選單\n"
     "[##]            移到該選項\n"
     "[F][U]          將文章寄回 Internet 郵箱/"
     "將文章 uuencode 後寄回郵箱\n");
    if (level >= MANAGER) {
    outs("\n\033[36m【 板主專用鍵 】\033[m\n"
         "[H]             切換為 公開/會員/板主 才能閱\讀\n"
         "[n/g/G]         收錄精華文章/開闢目錄/建立連線\n"
         "[m/d/D]         移動/刪除文章/刪除一個範圍的文章\n"
         "[f/T/e]         編輯標題符號/修改文章標題/內容\n"
         "[c/p/a]         拷貝/粘貼/附加文章\n"
         "[^P/^A]         粘貼/附加已用't'標記文章\n");
    }
    if (level >= SYSOP) {
    outs("\n\033[36m【 站長專用鍵 】\033[m\n"
         "[l]             建 symbolic link\n"
         "[N]             查詢檔名\n");
    }
    pressanykey();
}

static void
a_forward(char *path, fileheader_t * pitem, int mode)
{
    fileheader_t    fhdr;

    strlcpy(fhdr.filename, pitem->filename, sizeof(fhdr.filename));
    strlcpy(fhdr.title, pitem->title, sizeof(fhdr.title));
    switch (doforward(path, &fhdr, mode)) {
    case 0:
    outmsg(msg_fwd_ok);
    break;
    case -1:
    outmsg(msg_fwd_err1);
    break;
    case -2:
    outmsg(msg_fwd_err2);
    break;
    }
}

static void
a_additem(menu_t * pm, fileheader_t * myheader)
{
    char            buf[PATHLEN];

    setadir(buf, pm->path);
    if (append_record(buf, myheader, FHSZ) == -1)
    return;
    pm->now = pm->num++;

    if (pm->now >= pm->page + p_lines) {
    pm->page = pm->now - ((pm->page == 10000 && pm->now > p_lines / 2) ?
                  (p_lines / 2) : (pm->now % p_lines));
    }
    /* Ptt */
    strlcpy(pm->header[pm->now - pm->page].filename,
        myheader->filename,
        sizeof(pm->header[pm->now - pm->page].filename));
}

#define ADDITEM         0
#define ADDGROUP        1
#define ADDLINK         2

static void
a_newitem(menu_t * pm, int mode)
{
    char    *mesg[3] = {
    "[新增文章] 請輸入標題:",    /* ADDITEM */
    "[新增目錄] 請輸入標題:",    /* ADDGROUP */
    "請輸入標題:"        /* ADDLINK */
    };

    char            fpath[PATHLEN], buf[PATHLEN], lpath[PATHLEN];
    fileheader_t    item;
    int             d;

    strlcpy(fpath, pm->path, sizeof(fpath));

    switch (mode) {
    case ADDITEM:
    stampfile(fpath, &item);
    strlcpy(item.title, "◇ ", sizeof(item.title));    /* A1BA */
    break;

    case ADDGROUP:
    stampdir(fpath, &item);
    strlcpy(item.title, "◆ ", sizeof(item.title));    /* A1BB */
    break;
    case ADDLINK:
    stamplink(fpath, &item);
    if (!getdata(b_lines - 2, 1, "新增連線:", buf, 61, DOECHO))
        return;
    if (invalid_pname(buf)) {
        unlink(fpath);
        outs("目的地路徑不合法!");
        igetch();
        return;
    }
    item.title[0] = 0;
    // XXX Is it alright?
    // ex: path= "PASSWD"
    for (d = 0; d <= 3; d++) {
        switch (d) {
        case 0:
        snprintf(lpath, sizeof(lpath), BBSHOME "/man/boards/%c/%s/%s",
            currboard[0], currboard, buf);
        break;
        case 1:
        snprintf(lpath, sizeof(lpath), "%s%s/%c/%s",
            BBSHOME, "/man/boards/", buf[0], buf);
        break;
        case 2:
        snprintf(lpath, sizeof(lpath), "%s%s%s",
            BBSHOME, "/", buf);
        break;
        case 3:
        snprintf(lpath, sizeof(lpath), "%s%s%s",
            BBSHOME, "/etc/", buf);
        break;
        }
        if (dashf(lpath)) {
        strlcpy(item.title, "☆ ", sizeof(item.title));    /* A1B3 */
        break;
        } else if (dashd(lpath)) {
        strlcpy(item.title, "★ ", sizeof(item.title));    /* A1B4 */
        break;
        }
        if (!HAS_PERM(PERM_BBSADM) && d == 1)
        break;
    }

    if (!item.title[0]) {
        unlink(fpath);
        outs("目的地路徑不合法!");
        igetch();
        return;
    }
    }

    if (!getdata(b_lines - 1, 1, mesg[mode], &item.title[3], 55, DOECHO)) {
    if (mode == ADDGROUP)
        rmdir(fpath);
    else
        unlink(fpath);
    return;
    }
    switch (mode) {
    case ADDITEM:
    if (vedit(fpath, 0, NULL) == -1) {
        unlink(fpath);
        pressanykey();
        return;
    }
    break;
    case ADDLINK:
    unlink(fpath);
    if (symlink(lpath, fpath) == -1) {
        outs("無法建立 symbolic link");
        igetch();
        return;
    }
    break;
    }

    strlcpy(item.owner, cuser.userid, sizeof(item.owner));
    a_additem(pm, &item);
}

static void
a_pasteitem(menu_t * pm, int mode)
{
    char            newpath[PATHLEN];
    char            buf[PATHLEN];
    char            ans[2];
    int             i;
    fileheader_t    item;

    move(b_lines - 1, 1);
    if (copyfile[0]) {
    if (dashd(copyfile)) {
        for (i = 0; copyfile[i] && copyfile[i] == pm->path[i]; i++);
        if (!copyfile[i]) {
        outs("將目錄拷進自己的子目錄中,會造成無窮迴圈!");
        igetch();
        return;
        }
    }
    if (mode) {
        snprintf(buf, sizeof(buf),
             "確定要拷貝[%s]嗎(Y/N)?[N] ", copytitle);
        getdata(b_lines - 1, 1, buf, ans, sizeof(ans), LCECHO);
    } else
        ans[0] = 'y';
    if (ans[0] == 'y') {
        strlcpy(newpath, pm->path, sizeof(newpath));

        if (*copyowner) {
        char           *fname = strrchr(copyfile, '/');

        if (fname)
            strcat(newpath, fname);
        else
            return;
        if (access(pm->path, X_OK | R_OK | W_OK))
            mkdir(pm->path, 0755);
        memset(&item, 0, sizeof(fileheader_t));
        strlcpy(item.filename, fname + 1, sizeof(item.filename));
        memcpy(copytitle, "◎", 2);
        if (HAS_PERM(PERM_BBSADM))
            Link(copyfile, newpath);
        else {
                    Copy(copyfile, newpath);
        }
        } else if (dashf(copyfile)) {
        stampfile(newpath, &item);
        memcpy(copytitle, "◇", 2);
                Copy(copyfile, newpath);
        } else if (dashd(copyfile)) {
        stampdir(newpath, &item);
        memcpy(copytitle, "◆", 2);
        snprintf(buf, sizeof(buf),
             "/bin/cp -r %s/* %s/.D* %s", copyfile, copyfile,
             newpath);
        system(buf);
        } else {
        outs("無法拷貝!");
        igetch();
        return;
        }
        strlcpy(item.owner, *copyowner ? copyowner : cuser.userid,
            sizeof(item.owner));
        strlcpy(item.title, copytitle, sizeof(item.title));
        a_additem(pm, &item);
        copyfile[0] = '\0';
    }
    } else {
    outs("請先執行 copy 命令後再 paste");
    igetch();
    }
}

static void
a_appenditem(menu_t * pm, int isask)
{
    char            fname[PATHLEN];
    char            buf[ANSILINELEN];
    char            ans[2] = "y";
    FILE           *fp, *fin;

    move(b_lines - 1, 1);
    if (copyfile[0]) {
    if (dashf(copyfile)) {
        snprintf(fname, sizeof(fname), "%s/%s", pm->path,
            pm->header[pm->now - pm->page].filename);
        if (dashf(fname)) {
        if (isask) {
            snprintf(buf, sizeof(buf),
                 "確定要將[%s]附加於此嗎(Y/N)?[N] ", copytitle);
            getdata(b_lines - 2, 1, buf, ans, sizeof(ans), LCECHO);
        }
        if (ans[0] == 'y') {
            if ((fp = fopen(fname, "a+"))) {
            if ((fin = fopen(copyfile, "r"))) {
                memset(buf, '-', 74);
                buf[74] = '\0';
                fprintf(fp, "\n> %s <\n\n", buf);
                if (isask)
                getdata(b_lines - 1, 1,
                    "是否收錄簽名檔部份(Y/N)?[Y] ",
                    ans, sizeof(ans), LCECHO);
                while (fgets(buf, sizeof(buf), fin)) {
                if ((ans[0] == 'n') &&
                    !strcmp(buf, "--\n"))
                    break;
                fputs(buf, fp);
                }
                fclose(fin);
                copyfile[0] = '\0';
            }
            fclose(fp);
            }
        }
        } else {
        outs("檔案不得附加於此!");
        igetch();
        }
    } else {
        outs("不得附加整個目錄於檔案後!");
        igetch();
    }
    } else {
    outs("請先執行 copy 命令後再 append");
    igetch();
    }
}

static int
a_pastetagpost(menu_t * pm, int mode)
{
    fileheader_t    fhdr;
    boardheader_t  *bh = NULL;
    int             ans = 0, ent = 0, tagnum;
    char            title[TTLEN + 1] = "◇  ";
    char            dirname[200], buf[200];

    if (TagBoard == 0){
    sethomedir(dirname, cuser.userid);
    }
    else{
    bh = getbcache(TagBoard);
    setbdir(dirname, bh->brdname);
    }
    tagnum = TagNum;

    if (!tagnum)
    return ans;

    while (tagnum--) {
    EnumTagFhdr(&fhdr, dirname, ent++);
    if (TagBoard == 0) 
        sethomefile(buf, cuser.userid, fhdr.filename);
    else
        setbfile(buf, bh->brdname, fhdr.filename);

    if (dashf(buf)) {
        strncpy(title + 3, fhdr.title, TTLEN - 3);
        title[TTLEN] = '\0';
        a_copyitem(buf, title, 0, 0);
        if (mode) {
        mode--;
        a_pasteitem(pm, 0);
        } else
        a_appenditem(pm, 0);
        ++ans;
        UnTagger(tagnum);
    }
    }

    return ans;
}

static void
a_moveitem(menu_t * pm)
{
    fileheader_t   *tmp;
    char            newnum[4];
    int             num, max, min;
    char            buf[PATHLEN];
    int             fail;

    snprintf(buf, sizeof(buf), "請輸入第 %d 選項的新次序:", pm->now + 1);
    if (!getdata(b_lines - 1, 1, buf, newnum, sizeof(newnum), DOECHO))
    return;
    num = (newnum[0] == '$') ? 9999 : atoi(newnum) - 1;
    if (num >= pm->num)
    num = pm->num - 1;
    else if (num < 0)
    num = 0;
    setadir(buf, pm->path);
    min = num < pm->now ? num : pm->now;
    max = num > pm->now ? num : pm->now;
    tmp = (fileheader_t *) calloc(max + 1, FHSZ);

    fail = 0;
    if (get_records(buf, tmp, FHSZ, 1, min) != min)
    fail = 1;
    if (num > pm->now) {
    if (get_records(buf, &tmp[min], FHSZ, pm->now + 2, max - min) != max - min)
        fail = 1;
    if (get_records(buf, &tmp[max], FHSZ, pm->now + 1, 1) != 1)
        fail = 1;
    } else {
    if (get_records(buf, &tmp[min], FHSZ, pm->now + 1, 1) != 1)
        fail = 1;
    if (get_records(buf, &tmp[min + 1], FHSZ, num + 1, max - min) != max - min)
        fail = 1;
    }
    if (!fail)
    substitute_record(buf, tmp, FHSZ * (max + 1), 1);
    pm->now = num;
    free(tmp);
}

static void
a_delrange(menu_t * pm)
{
    char            fname[256];

    snprintf(fname, sizeof(fname), "%s/.DIR", pm->path);
    del_range(0, NULL, fname);
    pm->num = get_num_records(fname, FHSZ);
}

static void
a_delete(menu_t * pm)
{
    char            fpath[PATHLEN], buf[PATHLEN], cmd[PATHLEN];
    char            ans[4];
    fileheader_t    backup;

    snprintf(fpath, sizeof(fpath),
         "%s/%s", pm->path, pm->header[pm->now - pm->page].filename);
    setadir(buf, pm->path);

    if (pm->header[pm->now - pm->page].filename[0] == 'H' &&
    pm->header[pm->now - pm->page].filename[1] == '.') {
    getdata(b_lines - 1, 1, "您確定要刪除此精華區連線嗎(Y/N)?[N] ",
        ans, sizeof(ans), LCECHO);
    if (ans[0] != 'y')
        return;
    if (delete_record(buf, FHSZ, pm->now + 1) == -1)
        return;
    } else if (dashl(fpath)) {
    getdata(b_lines - 1, 1, "您確定要刪除此 symbolic link 嗎(Y/N)?[N] ",
        ans, sizeof(ans), LCECHO);
    if (ans[0] != 'y')
        return;
    if (delete_record(buf, FHSZ, pm->now + 1) == -1)
        return;
    unlink(fpath);
    } else if (dashf(fpath)) {
    getdata(b_lines - 1, 1, "您確定要刪除此檔案嗎(Y/N)?[N] ", ans,
        sizeof(ans), LCECHO);
    if (ans[0] != 'y')
        return;
    if (delete_record(buf, FHSZ, pm->now + 1) == -1)
        return;

    setbpath(buf, "deleted");
    stampfile(buf, &backup);
    strlcpy(backup.owner, cuser.userid, sizeof(backup.owner));
    strlcpy(backup.title,
        pm->header[pm->now - pm->page].title + 2,
        sizeof(backup.title));

    snprintf(cmd, sizeof(cmd),
         "mv -f %s %s", fpath, buf);
    system(cmd);
    setbdir(buf, "deleted");
    append_record(buf, &backup, sizeof(backup));
    } else if (dashd(fpath)) {
    getdata(b_lines - 1, 1, "您確定要刪除整個目錄嗎(Y/N)?[N] ", ans,
        sizeof(ans), LCECHO);
    if (ans[0] != 'y')
        return;
    if (delete_record(buf, FHSZ, pm->now + 1) == -1)
        return;

    setapath(buf, "deleted");
    stampdir(buf, &backup);

    snprintf(cmd, sizeof(cmd),
         "rm -rf %s;/bin/mv -f %s %s", buf, fpath, buf);
    system(cmd);

    strlcpy(backup.owner, cuser.userid, sizeof(backup.owner));
    strlcpy(backup.title,
        pm->header[pm->now - pm->page].title + 2,
        sizeof(backup.title));
    setapath(buf, "deleted");
    setadir(buf, buf);
    append_record(buf, &backup, sizeof(backup));
    } else {            /* Ptt 損毀的項目 */
    getdata(b_lines - 1, 1, "您確定要刪除此損毀的項目嗎(Y/N)?[N] ",
        ans, sizeof(ans), LCECHO);
    if (ans[0] != 'y')
        return;
    if (delete_record(buf, FHSZ, pm->now + 1) == -1)
        return;
    }
    pm->num--;
}

static void
a_newtitle(menu_t * pm)
{
    char            buf[PATHLEN];
    fileheader_t    item;

    memcpy(&item, &pm->header[pm->now - pm->page], FHSZ);
    strlcpy(buf, item.title + 3, sizeof(buf));
    if (getdata_buf(b_lines - 1, 1, "新標題:", buf, 60, DOECHO)) {
    strlcpy(item.title + 3, buf, sizeof(item.title) - 3);
    setadir(buf, pm->path);
    substitute_record(buf, &item, FHSZ, pm->now + 1);
    }
}
static void
a_hideitem(menu_t * pm)
{
    fileheader_t   *item = &pm->header[pm->now - pm->page];
    char            buf[PATHLEN];
    if (item->filemode & FILE_BM) {
    item->filemode &= ~FILE_BM;
    item->filemode &= ~FILE_HIDE;
    } else if (item->filemode & FILE_HIDE)
    item->filemode |= FILE_BM;
    else
    item->filemode |= FILE_HIDE;
    setadir(buf, pm->path);
    substitute_record(buf, item, FHSZ, pm->now + 1);
}
static void
a_editsign(menu_t * pm)
{
    char            buf[PATHLEN];
    fileheader_t    item;

    memcpy(&item, &pm->header[pm->now - pm->page], FHSZ);
    snprintf(buf, sizeof(buf), "%c%c", item.title[0], item.title[1]);
    if (getdata_buf(b_lines - 1, 1, "符號", buf, 5, DOECHO)) {
    item.title[0] = buf[0] ? buf[0] : ' ';
    item.title[1] = buf[1] ? buf[1] : ' ';
    item.title[2] = buf[2] ? buf[2] : ' ';
    setadir(buf, pm->path);
    substitute_record(buf, &item, FHSZ, pm->now + 1);
    }
}

static void
a_showname(menu_t * pm)
{
    char            buf[PATHLEN];
    int             len;
    int             i;
    int             sym;

    move(b_lines - 1, 1);
    snprintf(buf, sizeof(buf),
         "%s/%s", pm->path, pm->header[pm->now - pm->page].filename);
    if (dashl(buf)) {
    prints("此 symbolic link 名稱為 %s\n",
           pm->header[pm->now - pm->page].filename);
    if ((len = readlink(buf, buf, PATHLEN - 1)) >= 0) {
        buf[len] = '\0';
        for (i = 0; BBSHOME[i] && buf[i] == BBSHOME[i]; i++);
        if (!BBSHOME[i] && buf[i] == '/') {
        if (HAS_PERM(PERM_BBSADM))
            sym = 1;
        else {
            sym = 0;
            for (i++; BBSHOME "/man"[i] && buf[i] == BBSHOME "/man"[i];
             i++);
            if (!BBSHOME "/man"[i] && buf[i] == '/')
            sym = 1;
        }
        if (sym) {
            pressanykey();
            move(b_lines - 1, 1);
            prints("此 symbolic link 指向 %s\n", &buf[i + 1]);
        }
        }
    }
    } else if (dashf(buf))
    prints("此文章名稱為 %s", pm->header[pm->now - pm->page].filename);
    else if (dashd(buf))
    prints("此目錄名稱為 %s", pm->header[pm->now - pm->page].filename);
    else
    outs("此項目已損毀, 建議將其刪除!");
    pressanykey();
}

#if 0
static char    *a_title;

static void
atitle()
{
    showtitle("精華文章", a_title);
    outs("[←]離開 [→]閱\讀 [^P]發表文章 [b]備忘錄 [d]刪除 [q]精華區 "
     "[TAB]文摘 [h]elp\n\033[7m  編號   日 期  作  者       "
     "文  章  標  題\033[m");
}
#endif

static int
isvisible_man(menu_t * me)
{
    fileheader_t   *fhdr = &me->header[me->now - me->page];
    if (me->level < MANAGER && ((fhdr->filemode & FILE_BM) ||
                ((fhdr->filemode & FILE_HIDE) &&
                 /* board friend only effact when
                  * in board reading mode             */
                 (currstat == ANNOUNCE ||
                  hbflcheck(currbid, currutmp->uid))
                )))
    return 0;
    return 1;
}
int
a_menu(char *maintitle, char *path, int lastlevel)
{
    static char     Fexit;  // 用來跳出 recursion
    menu_t          me;
    char            fname[PATHLEN];
    int             ch, returnvalue = FULLUPDATE;

    trans_buffer[0] = 0;

    Fexit = 0;
    me.header = (fileheader_t *) calloc(p_lines, FHSZ);
    me.path = path;
    strlcpy(me.mtitle, maintitle, sizeof(me.mtitle));
    setadir(fname, me.path);
    me.num = get_num_records(fname, FHSZ);

    /* 精華區-tree 中部份結構屬於 cuser ==> BM */

    if (!(me.level = lastlevel)) {
    char           *ptr;

    if ((ptr = strrchr(me.mtitle, '[')))
        me.level = is_BM(ptr + 1);
    }
    me.page = 9999;
    me.now = 0;
    for (;;) {
    if (me.now >= me.num)
        me.now = me.num - 1;
    if (me.now < 0)
        me.now = 0;

    if (me.now < me.page || me.now >= me.page + p_lines) {
        me.page = me.now - ((me.page == 10000 && me.now > p_lines / 2) ?
                (p_lines / 2) : (me.now % p_lines));
        a_showmenu(&me);
    }
    ch = cursor_key(2 + me.now - me.page, 0);

    if (ch == 'q' || ch == 'Q' || ch == KEY_LEFT)
        break;

    if (ch >= '1' && ch <= '9') {
        if ((ch = search_num(ch, me.num)) != -1)
        me.now = ch;
        me.page = 10000;
        continue;
    }
    switch (ch) {
    case KEY_UP:
    case 'k':
        if (--me.now < 0)
        me.now = me.num - 1;
        break;

    case KEY_DOWN:
    case 'j':
        if (++me.now >= me.num)
        me.now = 0;
        break;

    case KEY_PGUP:
    case Ctrl('B'):
        if (me.now >= p_lines)
        me.now -= p_lines;
        else if (me.now > 0)
        me.now = 0;
        else
        me.now = me.num - 1;
        break;

    case ' ':
    case KEY_PGDN:
    case Ctrl('F'):
        if (me.now < me.num - p_lines)
        me.now += p_lines;
        else if (me.now < me.num - 1)
        me.now = me.num - 1;
        else
        me.now = 0;
        break;

    case '0':
        me.now = 0;
        break;
    case '?':
    case '/':
        me.now = a_searchtitle(&me, ch == '?');
        me.page = 9999;
        break;
    case '$':
        me.now = me.num - 1;
        break;
    case 'h':
        a_showhelp(me.level);
        me.page = 9999;
        break;

    case Ctrl('I'):
        t_idle();
        me.page = 9999;
        break;

    case 'e':
    case 'E':
        snprintf(fname, sizeof(fname),
             "%s/%s", path, me.header[me.now - me.page].filename);
        if (dashf(fname) && me.level >= MANAGER) {
        *quote_file = 0;
        if (vedit(fname, NA, NULL) != -1) {
            char            fpath[200];
            fileheader_t    fhdr;

            strlcpy(fpath, path, sizeof(fpath));
            stampfile(fpath, &fhdr);
            unlink(fpath);
            Rename(fname, fpath);
            strlcpy(me.header[me.now - me.page].filename,
                fhdr.filename,
                sizeof(me.header[me.now - me.page].filename));
            strlcpy(me.header[me.now - me.page].owner,
                cuser.userid,
                sizeof(me.header[me.now - me.page].owner));
            setadir(fpath, path);
            substitute_record(fpath, me.header + me.now - me.page,
                      sizeof(fhdr), me.now + 1);
        }
        me.page = 9999;
        }
        break;

    case 'c':
        if (me.now < me.num) {
        if (!isvisible_man(&me))
            break;

        snprintf(fname, sizeof(fname), "%s/%s", path,
             me.header[me.now - me.page].filename);

        /* XXX: dirty fix
           應該要改成如果發現該目錄裡面有隱形目錄的話才拒絕.
           不過這樣的話須要整個搜一遍, 而且目前判斷該資料是目錄
           還是檔案竟然是用 fstat(2) 而不是直接存在 .DIR 內 |||b
           須等該資料寫入 .DIR 內再 implement才有效率.
         */
        if( !lastlevel && !HAS_PERM(PERM_SYSOP) &&
            !is_BM_cache(currbid) && dashd(fname) )
            vmsg("只有板主才可以拷貝目錄唷!");
        else
            a_copyitem(fname, me.header[me.now - me.page].title, 0, 1);
        me.page = 9999;
        break;
        }
    case '\n':
    case '\r':
    case KEY_RIGHT:
    case 'r':
        if (me.now < me.num) {
        fileheader_t   *fhdr = &me.header[me.now - me.page];
        if (!isvisible_man(&me))
            break;
#ifdef DEBUG
        vmsg("%s/%s", &path[11], fhdr->filename);;
#endif
        snprintf(fname, sizeof(fname), "%s/%s", path, fhdr->filename);
        if (*fhdr->filename == 'H' && fhdr->filename[1] == '.') {
          vmsg("不再支援 gopher mode, 請使用瀏覽器直接瀏覽");
          vmsg("gopher://%s/1/",fhdr->filename+2);
        } else if (dashf(fname)) {
            int             more_result;

            while ((more_result = more(fname, YEA))) {
            /* Ptt 範本精靈 plugin */
            if (currstat == EDITEXP || currstat == OSONG) {
                char            ans[4];

                move(22, 0);
                clrtoeol();
                getdata(22, 1,
                    currstat == EDITEXP ?
                    "要把範例 Plugin 到文章嗎?[y/N]" :
                    "確定要點這首歌嗎?[y/N]",
                    ans, sizeof(ans), LCECHO);
                if (ans[0] == 'y') {
                strlcpy(trans_buffer,
                    fname, sizeof(trans_buffer));
                Fexit = 1;
                if (currstat == OSONG) {
                    /* XXX: 只選歌未點歌可灌排行榜 */
                    char    buf[128];
                    snprintf(buf, sizeof(buf),
                         "%s\n", fhdr->title);
                    log_file(FN_USSONG, buf, 1);
                }
                free(me.header);
                return FULLUPDATE;
                }
            }
            if (more_result == 1) {
                if (--me.now < 0) {
                me.now = 0;
                break;
                }
            } else if (more_result == 3) {
                if (++me.now >= me.num) {
                me.now = me.num - 1;
                break;
                }
            } else
                break;
            if (!isvisible_man(&me))
                break;
            snprintf(fname, sizeof(fname), "%s/%s", path,
                 me.header[me.now - me.page].filename);
            if (!dashf(fname))
                break;
            }
        } else if (dashd(fname)) {
            a_menu(me.header[me.now - me.page].title, fname, me.level);
            /* Ptt  強力跳出recursive */
            if (Fexit) {
            free(me.header);
            return FULLUPDATE;
            }
        }
        me.page = 9999;
        }
        break;

    case 'F':
    case 'U':
        snprintf(fname, sizeof(fname),
             "%s/%s", path, me.header[me.now - me.page].filename);
        if (me.now < me.num && HAS_PERM(PERM_BASIC) && dashf(fname)) {
        a_forward(path, &me.header[me.now - me.page], ch /* == 'U' */ );
        /* By CharlieL */
        } else
        outmsg("無法轉寄此項目");

        me.page = 9999;
        refresh();
        sleep(1);
        break;

#ifdef BLOG
    case 'b':
        if( !HAS_PERM(PERM_SYSOP) && !is_BM_cache(currbid) )
        vmsg("只有板主才可以用唷!");
        else{
        char    genbuf[128];
        snprintf(genbuf, sizeof(genbuf),
             "bin/builddb.pl -f -n %d %s", me.now, currboard);
        system(genbuf);
        vmsg("資料更新完成");
        }
        me.page = 9999;
        break;

    case 'B':
        if( !HAS_PERM(SYSOP) && !is_BM_cache(currbid) )
        vmsg("只有板主才可以用唷!");
        else
        BlogMain(me.now);
        me.page = 9999;

        break;
#endif
    }

    if (me.level >= MANAGER) {
        int             page0 = me.page;

        switch (ch) {
        case 'n':
        a_newitem(&me, ADDITEM);
        me.page = 9999;
        break;
        case 'g':
        a_newitem(&me, ADDGROUP);
        me.page = 9999;
        break;
        case 'p':
        a_pasteitem(&me, 1);
        me.page = 9999;
        break;
        case 'f':
        a_editsign(&me);
        me.page = 9999;
        break;
        case Ctrl('P'):
        a_pastetagpost(&me, -1);
        returnvalue = DIRCHANGED;
        me.page = 9999;
        break;
        case Ctrl('A'):
        a_pastetagpost(&me, 1);
        returnvalue = DIRCHANGED;
        me.page = 9999;
        break;
        case 'a':
        a_appenditem(&me, 1);
        me.page = 9999;
        break;
        default:
        me.page = page0;
        break;
        }

        if (me.num)
        switch (ch) {
        case 'm':
            a_moveitem(&me);
            me.page = 9999;
            break;

        case 'D':
            /* Ptt me.page = -1; */
            a_delrange(&me);
            me.page = 9999;
            break;
        case 'd':
            a_delete(&me);
            me.page = 9999;
            break;
        case 'H':
            a_hideitem(&me);
            me.page = 9999;
            break;
        case 'T':
            a_newtitle(&me);
            me.page = 9999;
            break;
        }
    }
    if (me.level == SYSOP) {
        switch (ch) {
        case 'l':
        a_newitem(&me, ADDLINK);
        me.page = 9999;
        break;
        case 'N':
        a_showname(&me);
        me.page = 9999;
        break;
        }
    }
    }
    free(me.header);
    return returnvalue;
}

static char    *mytitle = BBSNAME "佈告欄";

int
Announce()
{
    setutmpmode(ANNOUNCE);
    a_menu(mytitle, "man",
       ((HAS_PERM(PERM_SYSOP) || HAS_PERM(PERM_ANNOUNCE)) ? SYSOP :
        NOBODY));
    return 0;
}

#ifdef BLOG
#include <mysql/mysql.h>
void BlogMain(int num)
{
    int     oldmode = currutmp->mode;
    char    genbuf[128], exit = 0;

    //setutmpmode(BLOGGING); /* will crash someone using old program  */
    sprintf(genbuf, "%s的部落格", currboard);
    showtitle("部落格", genbuf);
    while( !exit ){
    move(1, 0);
    prints("請選擇您要執行的重作:\n"
           "0.回到上一層\n"
           "1.製作部落格樣板格式\n"
           "  使用新的 config 目錄下樣板資料\n"
           "  通常在第一次使用部落格或樣板更新的時候使用\n"
           "\n"
           "2.重新製作部落格\n"
           "  只在部落格資料整個亂掉的時候才使用\n"
           "\n"
           "3.將本文加入部落格\n"
           "  將游標所在位置的文章加入部落格\n"
           "\n"
           "4.刪除迴響\n"
           "\n"
           "5.刪除一篇部落格\n"
           "\n"
           "C.建立部落格目錄 (您只有第一次時需要)\n"
           );
    switch( getans("請選擇(0-5,C)?[0]") ){
    case '1':
        snprintf(genbuf, sizeof(genbuf),
             "bin/builddb.pl -c %s", currboard);
        system(genbuf);
        break;
    case '2':
        snprintf(genbuf, sizeof(genbuf),
             "bin/builddb.pl -a %s", currboard);
        system(genbuf);
        break;
    case '3':
        snprintf(genbuf, sizeof(genbuf),
             "bin/builddb.pl -f -n %d %s", num, currboard);
        system(genbuf);
        break;
    case '4':{
        char    hash[35];
        int     i;
        getdata(16, 0, "請輸入該篇的雜湊值: ",
            hash, sizeof(hash), DOECHO);
        for( i = 0 ; hash[i] != 0 ; ++i ) /* 前面用 getdata() 保證有 \0 */
        if( !islower(hash[i]) && !isnumber(hash[i]) )
            break;
        if( i != 32 ){
        vmsg("輸入錯誤");
        break;
        }
        if( hash[0] != 0 && 
        getans("請確定刪除(Y/N)?[N] ") == 'y' ){
        MYSQL   mysql;
        char    cmd[256];
        
        sprintf(cmd, "delete from comment where "
            "hash='%s'&&brdname='%s'", hash, currboard);
#ifdef DEBUG
        vmsg(cmd);
#endif
        if( !(!mysql_init(&mysql) ||
              !mysql_real_connect(&mysql, BLOGDB_HOST, BLOGDB_USER,
                      BLOGDB_PASSWD, BLOGDB_DB, 
                      BLOGDB_PORT, BLOGDB_SOCK, 0) ||
              mysql_query(&mysql, cmd)) )
            vmsg("資料刪除完成");
        else
            vmsg(
#ifdef DEBUG
             mysql_error(&mysql)
#else
             "database internal error"
#endif
             );
        mysql_close(&mysql);
        }
    }
        break;
        
    case '5': {
        char    date[9];
        getdata(16, 0, "請輸入該篇的日期(yyyymmdd): ",
            date, sizeof(date), DOECHO);
        snprintf(genbuf, sizeof(genbuf),
             "bin/builddb.pl -D %s %s", date, currboard);
        system(genbuf);
    }
        break;

    case 'C': case 'c': {
        fileheader_t item;
        char    fpath[PATHLEN], adir[PATHLEN], buf[256];
        sprintf(fpath, "man/boards/%c/%s", currboard[0], currboard);
        stampdir(fpath, &item);
        strlcpy(item.title, "◆ Blog", sizeof(item.title));
        strlcpy(item.owner, cuser.userid, sizeof(item.owner));

        sprintf(adir, "man/boards/%c/%s/.DIR", currboard[0], currboard);
        append_record(adir, &item, FHSZ);

        snprintf(buf, sizeof(buf),
             "cp -R etc/Blog.Default/.DIR etc/Blog.Default/* %s/",
             fpath);
        system(buf);

        more("etc/README.BLOG", YEA);
    }
        break;

    default:
        exit = 1;
        break;
    }
    if( !exit )
        vmsg("部落格完成");
    }
    currutmp->mode = oldmode;
    pressanykey();
}
#endif