/* $Id: bbs.c,v 1.24 2002/05/25 17:49:48 ptt Exp $ */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <time.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include "config.h"
#include "pttstruct.h"
#include "perm.h"
#include "modes.h"
#include "common.h"
#include "proto.h"
extern struct bcache_t *brdshm;
/*
static int g_board_names(boardheader_t *fhdr) {
AddNameList(fhdr->brdname);
return 0;
}
*/
extern userec_t cuser;
extern void touchdircache(int bid);
extern int TagNum;
extern time_t now;
extern char fromhost[];
static void mail_by_link(char* owner, char* title, char* path) {
char genbuf[200];
fileheader_t mymail;
sprintf(genbuf,BBSHOME"/home/%c/%s", cuser.userid[0], cuser.userid);
stampfile(genbuf, &mymail);
strcpy(mymail.owner, owner);
sprintf(mymail.title, title);
unlink(genbuf);
Link(path, genbuf);
sprintf(genbuf,BBSHOME"/home/%c/%s/.DIR",cuser.userid[0],cuser.userid);
append_record(genbuf, &mymail, sizeof(mymail));
}
extern int usernum;
void anticrosspost() {
char buf[200];
sprintf(buf,
"\033[1;33;46m%s \033[37;45mcross post 文章 \033[37m %s\033[m",
cuser.userid, ctime(&now));
log_file("etc/illegal_money", buf);
post_violatelaw(cuser.userid, "Ptt系統警察", "Cross-post", "罰單處份");
cuser.userlevel |= PERM_VIOLATELAW;
cuser.vl_count ++;
mail_by_link("Ptt警察部隊", "Cross-Post罰單",
BBSHOME "/etc/crosspost.txt");
passwd_update(usernum, &cuser);
exit(0);
}
/* Heat CharlieL*/
int save_violatelaw() {
char buf[128], ok[3];
setutmpmode(VIOLATELAW);
clear();
stand_title("繳罰單中心");
if(!(cuser.userlevel & PERM_VIOLATELAW)) {
mprints(22, 0, "\033[1;31m你無聊啊? 你又沒有被開罰單~~\033[m");
pressanykey();
return 0;
}
reload_money();
if(cuser.money < (int)cuser.vl_count*1000) {
sprintf(buf, "\033[1;31m這是你第 %d 次違反本站法規"
"必須繳出 %d $Ptt ,你只有 %d 元, 錢不夠啦!!\033[m",
(int)cuser.vl_count, (int)cuser.vl_count * 1000, cuser.money);
mprints(22, 0, buf);
pressanykey();
return 0;
}
move(5,0);
prints("\033[1;37m你知道嗎? 因為你的違法 "
"已經造成很多人的不便\033[m\n");
prints("\033[1;37m你是否確定以後不會再犯了?\033[m\n");
if(!getdata(10,0,"確定嗎?[y/n]:", ok, sizeof(ok), LCECHO) ||
ok[0] == 'n' || ok[0] == 'N') {
mprints(22,0,"\033[1;31m等你想通了再來吧!! "
"我相信你不會知錯不改的~~~\033[m");
pressanykey();
return 0;
}
sprintf(buf, "這是你第 %d 次違法 必須繳出 %d $Ptt",
cuser.vl_count, cuser.vl_count*1000);
mprints(11,0,buf);
if(!getdata(10, 0, "要付錢[y/n]:", ok, sizeof(ok), LCECHO) ||
ok[0] == 'N' || ok[0] == 'n') {
mprints(22,0, "\033[1;31m 嗯 存夠錢 再來吧!!!\033[m");
pressanykey();
return 0;
}
demoney(-1000*cuser.vl_count);
cuser.userlevel &= (~PERM_VIOLATELAW);
passwd_update(usernum, &cuser);
return 0;
}
/*
void make_blist() {
CreateNameList();
apply_boards(g_board_names);
}
*/
extern int currbid;
extern char currBM[];
extern int currmode;
extern char currboard[];
static time_t board_note_time;
static char *brd_title;
void set_board() {
boardheader_t *bp;
bp = getbcache(currbid);
board_note_time = bp->bupdate;
brd_title = bp->BM;
if(brd_title[0] <= ' ')
brd_title = "徵求中";
sprintf(currBM, "板主:%s", brd_title);
brd_title = ((bp->bvote != 2 && bp->bvote) ? "本看板進行投票中" :
bp->title + 7);
currmode = (currmode & (MODE_DIRTY | MODE_MENU)) | MODE_STARTED ;
if (HAS_PERM(PERM_ALLBOARD) || is_BM(bp->BM))
currmode = currmode | MODE_BOARD | MODE_POST;
else if(haspostperm(currboard))
currmode |= MODE_POST;
}
static void readtitle() {
showtitle(currBM, brd_title);
outs("[←]離開 [→]閱\讀 [^P]發表文章 [b]備忘錄 [d]刪除 [z]精華區 "
"[TAB]文摘 [h]elp\n\033[7m 編號 日 期 作 者 文 章 標 題"
" \033[m");
}
extern int brc_num;
extern int brc_list[];
extern char currtitle[TTLEN + 1];
extern int Tagger();
static void readdoent(int num, fileheader_t *ent) {
int type;
char *mark, *title, color,
special=0;
if(ent->recommend>9 || ent->recommend <0 ) ent->recommend=0; //Ptt:暫時
type = brc_unread(ent->filename,brc_num,brc_list) ? '+' : ' ';
if((currmode & MODE_BOARD) && (ent->filemode & FILE_DIGEST))
type = (type == ' ') ? '*' : '#';
else if(currmode & MODE_BOARD || HAS_PERM(PERM_LOGINOK)) {
if(ent->filemode & FILE_MARKED)
type = (type == ' ') ? 'm' : 'M';
else if (TagNum && !Tagger(atoi(ent->filename + 2), 0, TAG_NIN))
type = 'D';
else if (ent->filemode & FILE_SOLVED)
type = 's';
}
title = subject(mark = ent->title);
if(title == mark)
color = '1', mark = "□";
else
color = '3', mark = "R:";
if(title[47])
strcpy(title + 44, " …"); /* 把多餘的 string 砍掉 */
if(title[0]=='[' && title[3]==']' ) special=1;
if(strncmp(currtitle, title, TTLEN))
prints("%6d %c\033[1;32m%c\033[m%-6s%-13.12s%s "
"\033[1m%.*s\033[m%s\n", num, type,
ent->recommend?ent->recommend+'0':' ',
ent->date, ent->owner, mark,
special?6:0, title, special?title+6:title);
else
prints("%6d %c\033[1;32m%c\033[m%-6s%-13.12s\033[1;3%cm%s "
"%s\033[m\n", num, type,
ent->recommend?ent->recommend+'0':' ',
ent->date, ent->owner, color, mark,
title);
}
extern char currfile[];
int cmpfilename(fileheader_t *fhdr) {
return (!strcmp(fhdr->filename, currfile));
}
extern unsigned char currfmode;
int cmpfmode(fileheader_t *fhdr) {
return (fhdr->filemode & currfmode);
}
extern char currowner[];
int cmpfowner(fileheader_t *fhdr) {
return !strcasecmp(fhdr->owner, currowner);
}
extern char *err_bid;
extern userinfo_t *currutmp;
int whereami(int ent, fileheader_t *fhdr, char *direct) {
boardheader_t *bh, *p[32], *root;
int i,j;
if(!currutmp->brc_id) return 0;
move(1,0);
clrtobot();
bh=getbcache(currutmp->brc_id);
root=getbcache(1);
p[0]=bh;
for(i=0;i<31 && p[i]->parent!=root && p[i]->parent;i++)
p[i+1]=p[i]->parent;
j=i;
prints("我在哪?\n%-40.40s %.13s\n", p[j]->title+7, p[j]->BM);
for(j--;j>=0;j--)
prints("%*s %-13.13s %-37.37s %.13s\n", (i-j)*2, "",
p[j]->brdname, p[j]->title,
p[j]->BM);
pressanykey();
return FULLUPDATE;
}
static int substitute_check(fileheader_t *fhdr)
{
fileheader_t hdr;
char genbuf[100];
int num=0;
/* rocker.011018: 串接模式用reference增進效率 */
if ((currmode & MODE_SELECT) && (fhdr->money & FHR_REFERENCE))
{
num = fhdr->money & ~FHR_REFERENCE;
setbdir(genbuf, currboard);
get_record(genbuf, &hdr, sizeof (hdr), num);
/* 再這裡要check一下原來的dir裡面是不是有被人動過... */
if (strcmp (hdr.filename, fhdr->filename))
num = getindex(genbuf, fhdr->filename, sizeof(fileheader_t));
substitute_record(genbuf, fhdr, sizeof(*fhdr), num);
}
return num;
}
static int do_select(int ent, fileheader_t *fhdr, char *direct) {
char bname[20];
char bpath[60];
boardheader_t *bh;
struct stat st;
int i;
move(0, 0);
clrtoeol();
generalnamecomplete(MSG_SELECT_BOARD, bname, sizeof(bname),
brdshm->number,
completeboard_compar,
completeboard_permission,
completeboard_getname);
if(bname[0]=='\0' || !(i = getbnum(bname)))
return FULLUPDATE;
bh = getbcache(i);
if(!Ben_Perm(bh)) return FULLUPDATE;
strcpy(bname, bh->brdname);
currbid=i;
setbpath(bpath, bname);
if((*bname == '\0') || (stat(bpath, &st) == -1)) {
move(2, 0);
clrtoeol();
outs(err_bid);
return FULLUPDATE;
}
setutmpbid(currbid);
brc_initial(bname);
set_board();
setbdir(direct, currboard);
move(1, 0);
clrtoeol();
return NEWDIRECT;
}
/* ----------------------------------------------------- */
/* 改良 innbbsd 轉出信件、連線砍信之處理程序 */
/* ----------------------------------------------------- */
void outgo_post(fileheader_t *fh, char *board) {
FILE *foo;
if((foo = fopen("innd/out.bntp", "a"))) {
fprintf(foo, "%s\t%s\t%s\t%s\t%s\n", board,
fh->filename, cuser.userid, cuser.username, fh->title);
fclose(foo);
}
}
extern char *str_author1;
extern char *str_author2;
static void cancelpost(fileheader_t *fh, int by_BM) {
FILE *fin, *fout;
char *ptr, *brd;
fileheader_t postfile;
char genbuf[200];
char nick[STRLEN], fn1[STRLEN], fn2[STRLEN];
setbfile(fn1, currboard, fh->filename);
if((fin = fopen(fn1, "r"))) {
brd = by_BM ? "deleted" : "junk";
setbpath(fn2, brd);
stampfile(fn2, &postfile);
memcpy(postfile.owner, fh->owner, IDLEN + TTLEN + 10);
nick[0] = '\0';
while(fgets(genbuf, sizeof(genbuf), fin)) {
if (!strncmp(genbuf, str_author1, LEN_AUTHOR1) ||
!strncmp(genbuf, str_author2, LEN_AUTHOR2)) {
if((ptr = strrchr(genbuf, ')')))
*ptr = '\0';
if((ptr = (char *)strchr(genbuf, '(')))
strcpy(nick, ptr + 1);
break;
}
}
if((fout = fopen("innd/cancel.bntp", "a"))) {
fprintf(fout, "%s\t%s\t%s\t%s\t%s\n", currboard, fh->filename,
cuser.userid, nick, fh->title);
fclose(fout);
}
fclose(fin);
Rename(fn1, fn2);
setbdir(genbuf, brd);
append_record(genbuf, &postfile, sizeof(postfile));
}
}
extern char *str_reply;
extern char save_title[];
/* ----------------------------------------------------- */
/* 發表、回應、編輯、轉錄文章 */
/* ----------------------------------------------------- */
void do_reply_title(int row, char *title) {
char genbuf[200];
char genbuf2[4];
if(strncasecmp(title, str_reply, 4))
sprintf(save_title, "Re: %s", title);
else
strcpy(save_title, title);
save_title[TTLEN - 1] = '\0';
sprintf(genbuf, "採用原標題《%.60s》嗎?[Y] ", save_title);
getdata(row, 0, genbuf, genbuf2, 4, LCECHO);
if(genbuf2[0] == 'n' || genbuf2[0] == 'N')
getdata(++row, 0, "標題:", save_title, TTLEN, DOECHO);
}
static void do_unanonymous_post(char* fpath) {
fileheader_t mhdr;
char title[128];
char genbuf[200];
setbpath(genbuf, "UnAnonymous");
if(dashd(genbuf)) {
stampfile(genbuf, &mhdr);
unlink(genbuf);
Link(fpath, genbuf);
strcpy(mhdr.owner, cuser.userid);
strcpy(mhdr.title, save_title);
mhdr.filemode = 0;
setbdir(title, "UnAnonymous");
append_record(title, &mhdr, sizeof(mhdr));
}
}
extern char quote_file[];
extern char quote_user[];
extern int curredit;
extern unsigned int currbrdattr;
extern char currdirect[];
extern char *err_uid;
#ifdef NO_WATER_POST
static time_t last_post_time = 0;
static time_t water_counts = 0;
#endif
int local_article;
char real_name[IDLEN + 1];
static int do_general() {
fileheader_t postfile;
char fpath[80], buf[80];
int aborted, defanony, ifuseanony;
char genbuf[200],*owner;
boardheader_t *bp;
int islocal;
ifuseanony = 0;
bp = getbcache(currbid);
clear();
if(!(currmode & MODE_POST)) {
move(5, 10);
outs("對不起,您目前無法在此發表文章!");
pressanykey();
return FULLUPDATE;
}
#ifdef NO_WATER_POST
/* 三分鐘內最多發表五篇文章 */
if(currutmp->lastact - last_post_time < 60 * 3) {
if(water_counts >= 5) {
move(5, 10);
outs("對不起,您的文章太水囉,待會再post吧!小秘訣:可用'X'推薦文章");
pressanykey();
return FULLUPDATE;
}
} else {
last_post_time = currutmp->lastact;
water_counts = 0;
}
#endif
setbfile(genbuf, currboard, FN_POST_NOTE );
if(more(genbuf,NA) == -1)
more("etc/"FN_POST_NOTE , NA);
move(19,0);
prints("發表文章於【\033[33m %s\033[m 】 \033[32m%s\033[m 看板\n\n",
currboard, bp->title + 7);
if(quote_file[0])
do_reply_title(20, currtitle);
else {
getdata(21, 0, "標題:", save_title, TTLEN, DOECHO);
strip_ansi(save_title,save_title,0);
}
if(save_title[0] == '\0')
return FULLUPDATE;
curredit &= ~EDIT_MAIL;
curredit &= ~EDIT_ITEM;
setutmpmode(POSTING);
/* 未具備 Internet 權限者,只能在站內發表文章 */
if(HAS_PERM(PERM_INTERNET))
local_article = 0;
else
local_article = 1;
/* build filename */
setbpath(fpath, currboard);
stampfile(fpath, &postfile);
aborted = vedit(fpath, YEA, &islocal);
if(aborted == -1) {
unlink(fpath);
pressanykey();
return FULLUPDATE;
}
water_counts++; /* po成功 */
/* set owner to Anonymous , for Anonymous board */
#ifdef HAVE_ANONYMOUS
/* Ptt and Jaky */
defanony=currbrdattr & BRD_DEFAULTANONYMOUS;
if((currbrdattr & BRD_ANONYMOUS) &&
((strcmp(real_name,"r") && defanony) || (real_name[0] && !defanony))
) {
strcat(real_name,".");
owner = real_name;
ifuseanony=1;
} else
owner = cuser.userid;
#else
owner = cuser.userid;
#endif
/* 錢 */
aborted = (aborted > MAX_POST_MONEY * 2) ? MAX_POST_MONEY : aborted / 2;
postfile.money = aborted;
strcpy(postfile.owner, owner);
strcpy(postfile.title, save_title);
if(islocal) /* local save */
postfile.filemode = FILE_LOCAL;
setbdir(buf, currboard);
if(append_record(buf, &postfile, sizeof(postfile)) != -1) {
setbtotal(currbid);
if(currmode & MODE_SELECT)
append_record(currdirect,&postfile,sizeof(postfile));
if(!islocal && !(bp->brdattr & BRD_NOTRAN))
outgo_post(&postfile, currboard);
brc_addlist(postfile.filename);
if(!(currbrdattr & BRD_HIDE) &&
(!bp->level || (currbrdattr & BRD_POSTMASK))) {
setbpath(genbuf, ALLPOST);
stampfile(genbuf, &postfile);
unlink(genbuf);
/* jochang: boards may spread across many disk */
/* link doesn't work across device,
Link doesn't work if we have same-time-across-device posts,
we try symlink now */
{
/* we need absolute path for symlink */
char abspath[256]=BBSHOME"/";
strcat(abspath,fpath);
symlink(abspath,genbuf);
}
strcpy(postfile.owner, owner);
strcpy(postfile.title, save_title);
postfile.filemode = FILE_LOCAL;
setbdir(genbuf, ALLPOST);
if(append_record(genbuf, &postfile, sizeof(postfile)) != -1) {
setbtotal(getbnum(ALLPOST));
}
}
outs("順利貼出佈告,");
#ifdef MAX_POST_MONEY
aborted = (aborted > MAX_POST_MONEY) ? MAX_POST_MONEY : aborted;
#endif
if(strcmp(currboard, "Test") && !ifuseanony) {
prints("這是您的第 %d 篇文章。 稿酬 %d 銀。",
++cuser.numposts, aborted );
demoney(aborted);
passwd_update(usernum, &cuser); /* post 數 */
} else
outs("測試信件不列入紀錄,敬請包涵。");
/* 回應到原作者信箱 */
if(curredit & EDIT_BOTH) {
char *str, *msg = "回應至作者信箱";
if((str = strchr(quote_user, '.'))) {
if(
#ifndef USE_BSMTP
bbs_sendmail(fpath, save_title, str + 1)
#else
bsmtp(fpath, save_title, str + 1 ,0)
#endif
< 0)
msg = "作者無法收信";
} else {
sethomepath(genbuf, quote_user);
stampfile(genbuf, &postfile);
unlink(genbuf);
Link(fpath, genbuf);
strcpy(postfile.owner, cuser.userid);
strcpy(postfile.title, save_title);
postfile.filemode = FILE_BOTH;/* both-reply flag */
sethomedir(genbuf, quote_user);
if(append_record(genbuf, &postfile, sizeof(postfile)) == -1)
msg = err_uid;
}
outs(msg);
curredit ^= EDIT_BOTH;
}
if(currbrdattr & BRD_ANONYMOUS)
do_unanonymous_post(fpath);
}
pressanykey();
return FULLUPDATE;
}
int do_post() {
boardheader_t *bp;
bp = getbcache(currbid);
if(bp->brdattr & BRD_VOTEBOARD)
return do_voteboard();
else if(!(bp->brdattr & BRD_GROUPBOARD))
return do_general();
touchdircache(currbid);
return 0;
}
extern int b_lines;
extern int curredit;
static void do_generalboardreply(fileheader_t *fhdr){
char genbuf[200];
getdata(b_lines - 1, 0,
"▲ 回應至 (F)看板 (M)作者信箱 (B)二者皆是 (Q)取消?[F] ",
genbuf, 3, LCECHO);
switch(genbuf[0]) {
case 'm':
mail_reply(0, fhdr, 0);
case 'q':
break;
case 'b':
curredit = EDIT_BOTH;
default:
strcpy(currtitle, fhdr->title);
strcpy(quote_user, fhdr->owner);
do_post();
}
*quote_file = 0;
}
int getindex(char *fpath, char *fname, int size) {
int fd, now=0;
fileheader_t fhdr;
if((fd = open(fpath, O_RDONLY, 0)) != -1) {
while((read(fd, &fhdr, size) == size)) {
now++;
if(!strcmp(fhdr.filename,fname)) {
close(fd);
return now;
}
}
close(fd);
}
return 0;
}
int invalid_brdname(char *brd) {
register char ch;
ch = *brd++;
if(not_alnum(ch))
return 1;
while((ch = *brd++)) {
if(not_alnum(ch) && ch != '_' && ch != '-' && ch != '.')
return 1;
}
return 0;
}
static void do_reply(fileheader_t *fhdr) {
boardheader_t *bp;
bp = getbcache(currbid);
if (bp->brdattr & BRD_VOTEBOARD)
do_voteboardreply(fhdr);
else
do_generalboardreply(fhdr);
}
static int reply_post(int ent, fileheader_t *fhdr, char *direct) {
if(!(currmode & MODE_POST))
return DONOTHING;
setdirpath(quote_file, direct, fhdr->filename);
do_reply(fhdr);
*quote_file = 0;
return FULLUPDATE;
}
static int edit_post(int ent, fileheader_t *fhdr, char *direct) {
char fpath[80], fpath0[80];
char genbuf[200];
fileheader_t postfile;
boardheader_t *bp;
bp = getbcache(currbid);
if (!HAS_PERM(PERM_SYSOP) && (bp->brdattr & BRD_VOTEBOARD))
return DONOTHING;
if ((!HAS_PERM(PERM_SYSOP)) &&
strcmp(fhdr->owner, cuser.userid))
return DONOTHING;
setutmpmode(REEDIT);
setdirpath(genbuf, direct, fhdr->filename);
local_article = fhdr->filemode & FILE_LOCAL;
strcpy(save_title, fhdr->title);
/* rocker.011018: 這裡是不是該檢查一下修改文章後的money和原有的比較? */
if(vedit(genbuf, 0, NULL) != -1) {
setbpath(fpath, currboard);
stampfile(fpath, &postfile);
unlink(fpath);
setbfile(fpath0, currboard, fhdr->filename);
Rename(fpath0, fpath);
/* rocker.011018: fix 串接模式改文章後文章就不見的bug */
if ((currmode & MODE_SELECT) && (fhdr->money & FHR_REFERENCE))
{
fileheader_t hdr;
int num;
num = fhdr->money & ~FHR_REFERENCE;
setbdir(fpath0, currboard);
get_record(fpath0, &hdr, sizeof (hdr), num);
/* 再這裡要check一下原來的dir裡面是不是有被人動過... */
if (!strcmp (hdr.filename, fhdr->filename))
{
strcpy(hdr.filename, postfile.filename);
strcpy(hdr.title, save_title);
substitute_record(fpath0, &hdr, sizeof(hdr), num);
}
}
strcpy(fhdr->filename, postfile.filename);
strcpy(fhdr->title, save_title);
brc_addlist(postfile.filename);
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
/* rocker.011018: 順便更新一下cache */
touchdircache(currbid);
}
return FULLUPDATE;
}
extern crosspost_t postrecord;
#define UPDATE_USEREC (currmode |= MODE_DIRTY)
static int cross_post(int ent, fileheader_t *fhdr, char *direct) {
char xboard[20], fname[80], xfpath[80], xtitle[80], inputbuf[10];
fileheader_t xfile;
FILE *xptr;
int author = 0;
char genbuf[200];
char genbuf2[4];
boardheader_t *bp;
move(2, 0);
clrtoeol();
move(3, 0);
clrtoeol();
move(1, 0);
bp = getbcache(currbid);
if (bp && (bp->brdattr & BRD_VOTEBOARD))
return FULLUPDATE;
generalnamecomplete("轉錄本文章於看板:", xboard, sizeof(xboard),
brdshm->number,
completeboard_compar,
completeboard_permission,
completeboard_getname);
if(*xboard == '\0' || !haspostperm(xboard))
return FULLUPDATE;
if((ent = str_checksum(fhdr->title)) != 0 &&
ent == postrecord.checksum[0]) {
/* 檢查 cross post 次數 */
if(postrecord.times++ > MAX_CROSSNUM)
anticrosspost();
} else {
postrecord.times = 0;
postrecord.checksum[0] = ent;
}
ent = 1;
if(HAS_PERM(PERM_SYSOP) || !strcmp(fhdr->owner, cuser.userid)) {
getdata(2, 0, "(1)原文轉載 (2)舊轉錄格式?[1] ",
genbuf, 3, DOECHO);
if(genbuf[0] != '2') {
ent = 0;
getdata(2, 0, "保留原作者名稱嗎?[Y] ", inputbuf, 3, DOECHO);
if (inputbuf[0] != 'n' && inputbuf[0] != 'N') author = 1;
}
}
if(ent)
sprintf(xtitle, "[轉錄]%.66s", fhdr->title);
else
strcpy(xtitle, fhdr->title);
sprintf(genbuf, "採用原標題《%.60s》嗎?[Y] ", xtitle);
getdata(2, 0, genbuf, genbuf2, 4, LCECHO);
if(genbuf2[0] == 'n' || genbuf2[0] == 'N') {
if(getdata_str(2, 0, "標題:", genbuf, TTLEN, DOECHO,xtitle))
strcpy(xtitle, genbuf);
}
getdata(2, 0, "(S)存檔 (L)站內 (Q)取消?[Q] ", genbuf, 3, LCECHO);
if(genbuf[0] == 'l' || genbuf[0] == 's') {
int currmode0 = currmode;
currmode = 0;
setbpath(xfpath, xboard);
stampfile(xfpath, &xfile);
if(author)
strcpy(xfile.owner, fhdr->owner);
else
strcpy(xfile.owner, cuser.userid);
strcpy(xfile.title, xtitle);
if(genbuf[0] == 'l') {
xfile.filemode = FILE_LOCAL;
}
setbfile(fname, currboard, fhdr->filename);
// if(ent) {
xptr = fopen(xfpath, "w");
strcpy(save_title, xfile.title);
strcpy(xfpath, currboard);
strcpy(currboard, xboard);
write_header(xptr);
strcpy(currboard, xfpath);
fprintf(xptr, "※ [本文轉錄自 %s 看板]\n\n", currboard);
b_suckinfile(xptr, fname);
addsignature(xptr,0);
fclose(xptr);
/* Cross fs有問題
} else {
unlink(xfpath);
link(fname, xfpath);
}
*/
setbdir(fname, xboard);
append_record(fname, &xfile, sizeof(xfile));
bp = getbcache(getbnum(xboard));
if(!xfile.filemode && !(bp->brdattr && BRD_NOTRAN))
outgo_post(&xfile, xboard);
setbtotal(getbnum(xboard));
cuser.numposts++;
UPDATE_USEREC;
outs("文章轉錄完成");
pressanykey();
currmode = currmode0;
}
return FULLUPDATE;
}
static int read_post(int ent, fileheader_t *fhdr, char *direct) {
char genbuf[200];
int more_result;
if(fhdr->owner[0] == '-')
return DONOTHING;
setdirpath(genbuf, direct, fhdr->filename);
if((more_result = more(genbuf, YEA)) == -1)
return DONOTHING;
brc_addlist(fhdr->filename);
strncpy(currtitle, subject(fhdr->title), TTLEN);
strncpy(currowner, subject(fhdr->owner), IDLEN + 2);
switch (more_result) {
case 1:
return READ_PREV;
case 2:
return RELATE_PREV;
case 3:
return READ_NEXT;
case 4:
return RELATE_NEXT;
case 5:
return RELATE_FIRST;
case 6:
return FULLUPDATE;
case 7:
case 8:
if((currmode & MODE_POST)) {
strcpy(quote_file, genbuf);
do_reply(fhdr);
*quote_file = 0;
}
return FULLUPDATE;
case 9:
return 'A';
case 10:
return 'a';
case 11:
return '/';
case 12:
return '?';
}
outmsg("\033[34;46m 閱\讀文章 \033[31;47m (R/Y)\033[30m回信 \033[31m"
"(=[]<>)\033[30m相關主題 \033[31m(↑↓)\033[30m上下封 \033[31m(←)"
"\033[30m離開 \033[m");
switch(egetch()) {
case 'q':
case 'Q':
case KEY_LEFT:
break;
case ' ':
case KEY_RIGHT:
case KEY_DOWN:
case KEY_PGDN:
case 'n':
case Ctrl('N'):
return READ_NEXT;
case KEY_UP:
case 'p':
case Ctrl('P'):
case KEY_PGUP:
return READ_PREV;
case '=':
return RELATE_FIRST;
case ']':
case 't':
return RELATE_NEXT;
case '[':
return RELATE_PREV;
case '.':
case '>':
return THREAD_NEXT;
case ',':
case '<':
return THREAD_PREV;
case Ctrl('C'):
cal();
return FULLUPDATE;
break;
case Ctrl('I'):
t_idle();
return FULLUPDATE;
case 'y':
case 'r':
case 'R':
case 'Y':
if((currmode & MODE_POST)) {
strcpy(quote_file, genbuf);
do_reply(fhdr);
*quote_file = 0;
}
}
return FULLUPDATE;
}
/* ----------------------------------------------------- */
/* 採集精華區 */
/* ----------------------------------------------------- */
static int b_man() {
char buf[64];
setapath(buf, currboard);
if( (currmode & MODE_BOARD) || HAS_PERM(PERM_SYSOP) ){
char genbuf[128];
int fd;
sprintf(genbuf, "%s/.rebuild", buf);
if( (fd = open(genbuf, O_CREAT, 0640)) > 0 )
close(fd);
}
return a_menu(currboard, buf, HAS_PERM(PERM_ALLBOARD) ? 2 :
(currmode & MODE_BOARD ? 1 : 0));
}
#ifndef NO_GAMBLE
static int join_gamble(int ent, fileheader_t *fhdr, char *direct) {
ticket(currbid);
return FULLUPDATE;
}
static int hold_gamble(int ent, fileheader_t *fhdr, char *direct) {
char fn_ticket[128],fn_ticket_end[128],genbuf[128],
msg[256]="",yn[10]="";
int i;
FILE *fp=NULL;
if(!(currmode & MODE_BOARD)) return 0;
setbfile(fn_ticket, currboard, FN_TICKET);
setbfile(fn_ticket_end, currboard, FN_TICKET_END);
if(dashf(fn_ticket))
{
getdata(b_lines - 1, 0, "已經有舉辦賭盤, "
"是否要 [停止下注]?(N/y):", yn, 3, LCECHO);
if(yn[0]!='y') return FULLUPDATE;
rename(fn_ticket, fn_ticket_end);
return FULLUPDATE;
}
if(dashf(fn_ticket_end))
{
getdata(b_lines - 1, 0, "已經有舉辦賭盤, "
"是否要 [開獎]?(N/y):", yn, 3, LCECHO);
if(yn[0]!='y') return FULLUPDATE;
openticket(currbid);
return FULLUPDATE;
}
getdata(b_lines - 2, 0, "要舉辦賭盤 (N/y):", yn, 3, LCECHO);
if(yn[0]!='y') return FULLUPDATE;
getdata(b_lines - 1, 0, "賭什麼? 請輸入主題 (輸入後編輯內容):",
msg, 20, DOECHO);
if(msg[0]==0 ||
vedit(fn_ticket_end, NA, NULL)<0)
return FULLUPDATE;
clear();
showtitle("舉辦賭盤",BBSNAME);
setbfile(genbuf, currboard, FN_TICKET_ITEMS);
// sprintf(genbuf, "%s/"FN_TICKET_ITEMS, direct);
if(!(fp=fopen(genbuf,"w"))) return FULLUPDATE;
do
{
getdata(2, 0, "輸入彩票價格 (價格:10-10000):",yn,6, LCECHO);
i=atoi(yn);
} while( i<10 || i>10000);
fprintf(fp,"%d\n",i);
move(3,0);
sprintf(genbuf,"請到 %s 版 按'f'參與賭博!\n\n一張 %d Ptt幣, 這是%s的賭博\n",
currboard,
i, i<100 ? "小賭式" : i<500 ? "平民級":
i<1000 ?"貴族級" : i<5000 ?"富豪級" : "傾家蕩產");
strcat(msg, genbuf);
prints("請依次輸入彩票名稱, 需提供2~8項. (未滿八項, 輸入直接按enter)\n");
for(i=0; i<8; i++)
{
sprintf(yn, " %d)",i+1);
getdata(6+i, 0, yn, genbuf, 9, DOECHO);
if(!genbuf[0] && i>1)
break;
fprintf(fp,"%s\n",genbuf);
}
fclose(fp);
move(8+i,0);
prints("賭盤設定完成");
sprintf(genbuf,"[公告] %s 版 開始賭博!", currboard);
post_msg(currboard, genbuf, msg, cuser.userid);
post_msg("Record", genbuf+7, msg, "[馬路探子]");
/* Tim 控制CS, 以免正在玩的user把資料已經寫進來 */
rename(fn_ticket_end, fn_ticket); // 設定完才把檔名改過來
return FULLUPDATE;
}
#endif
static int cite_post(int ent, fileheader_t *fhdr, char *direct) {
char fpath[256];
char title[TTLEN + 1];
setbfile(fpath, currboard, fhdr->filename);
strcpy(title, "◇ ");
strncpy(title+3, fhdr->title, TTLEN-3);
title[TTLEN] = '\0';
a_copyitem(fpath, title, 0, 1);
b_man();
return FULLUPDATE;
}
int edit_title(int ent, fileheader_t *fhdr, char *direct) {
char genbuf[200];
fileheader_t tmpfhdr = *fhdr;
int dirty = 0;
if(currmode & MODE_BOARD || !strcmp(cuser.userid,fhdr->owner)) {
if(getdata(b_lines - 1, 0, "標題:", genbuf, TTLEN, DOECHO)) {
strcpy(tmpfhdr.title, genbuf);
dirty++;
}
}
if(HAS_PERM(PERM_SYSOP)) {
if(getdata(b_lines - 1, 0, "作者:", genbuf, IDLEN + 2, DOECHO)) {
strcpy(tmpfhdr.owner, genbuf);
dirty++;
}
if(getdata(b_lines - 1, 0, "日期:", genbuf, 6, DOECHO)) {
sprintf(tmpfhdr.date, "%.5s", genbuf);
dirty++;
}
}
if(currmode & MODE_BOARD || !strcmp(cuser.userid,fhdr->owner)) {
getdata(b_lines-1, 0, "確定(Y/N)?[n] ", genbuf, 3, DOECHO);
if((genbuf[0] == 'y' || genbuf[0] == 'Y') && dirty) {
*fhdr = tmpfhdr;
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
/* rocker.011018: 這裡應該改成用reference的方式取得原來的檔案 */
substitute_check(fhdr);
touchdircache(currbid);
}
return FULLUPDATE;
}
return DONOTHING;
}
extern unsigned int currstat;
static int solve_post(int ent, fileheader_t * fhdr, char *direct){
if (HAS_PERM(PERM_SYSOP)) {
fhdr->filemode ^= FILE_SOLVED;
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
touchdircache(currbid);
return PART_REDRAW;
}
return DONOTHING;
}
static int recommend_cancel(int ent, fileheader_t *fhdr, char *direct) {
char yn[5];
if (!(currmode & MODE_BOARD)) return DONOTHING;
getdata(b_lines-1, 0, "確定要推薦歸零(Y/N)?[n] ", yn, 5, LCECHO);
if(yn[0]!='y') return FULLUPDATE;
fhdr->recommend=0;
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
substitute_check(fhdr);
touchdircache(currbid);
return FULLUPDATE;
}
static int recommend(int ent, fileheader_t *fhdr, char *direct) {
struct tm *ptime=localtime(&now);
extern userec_t xuser;
char buf[200],path[200], yn[5];
if(!HAS_PERM(PERM_LOGINOK)) return DONOTHING;
if(fhdr->recommend>9 || fhdr->recommend<0 )// 暫時性的code 原來舊有值取消
fhdr->recommend=0;
if (!(currmode & MODE_BOARD) && getuser(cuser.userid) &&
now - xuser.recommend < 60 )
{
move(b_lines-1,0);
prints("離上次推薦時間太近囉, 請多花點時間仔細閱\讀文章!");
pressanykey();
return FULLUPDATE;
}
if(!getdata(b_lines-2, 0, "推薦語:",path,40,DOECHO) ||
!getdata(b_lines-1, 0, "確定要推薦, 請仔細考慮(Y/N)?[n] ", yn, 5,LCECHO)
|| yn[0]!='y') return FULLUPDATE;
sprintf(buf,
"\033[1;31m→\033[33m %s推薦:%s\033[m 來自: %s (%02d/%02d %02d:%02d)\n",
cuser.userid, path, fromhost,
ptime->tm_mon+1,ptime->tm_mday,ptime->tm_hour,ptime->tm_min) ;
setdirpath(path, direct, fhdr->filename);
log_file(path, buf);
if(fhdr->recommend<9)
{
fhdr->recommend++;
cuser.recommend=now;
passwd_update(usernum, &cuser);
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
substitute_check(fhdr);
touchdircache(currbid);
}
return FULLUPDATE;
}
static int mark_post(int ent, fileheader_t *fhdr, char *direct) {
if(!(currmode & MODE_BOARD)) return DONOTHING;
fhdr->filemode ^= FILE_MARKED;
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
substitute_check(fhdr);
touchdircache(currbid);
return PART_REDRAW;
}
extern char *msg_sure_ny;
int del_range(int ent, fileheader_t *fhdr, char *direct) {
char num1[8], num2[8];
int inum1, inum2;
/* rocker.011018: 串接模式下還是不允許刪除比較好 */
if(currmode & MODE_SELECT) {
outmsg("請先回到正常模式後再進行刪除...");
refresh();
/*safe_sleep(1);*/
return FULLUPDATE;
}
if((currstat != READING) || (currmode & MODE_BOARD)) {
getdata(1, 0, "[設定刪除範圍] 起點:", num1, 5, DOECHO);
inum1 = atoi(num1);
if(inum1 <= 0) {
outmsg("起點有誤");
refresh();
/*safe_sleep(1);*/
return FULLUPDATE;
}
getdata(1, 28, "終點:", num2, 5, DOECHO);
inum2 = atoi(num2);
if(inum2 < inum1) {
outmsg("終點有誤");
refresh();
/*safe_sleep(1);*/
return FULLUPDATE;
}
getdata(1, 48, msg_sure_ny, num1, 3, LCECHO);
if(*num1 == 'y') {
outmsg("處理中,請稍後...");
refresh();
if(currmode & MODE_SELECT) {
int fd,size = sizeof(fileheader_t);
char genbuf[100];
fileheader_t rsfh;
int i = inum1,now;
if(currstat == RMAIL)
sethomedir(genbuf, cuser.userid);
else
setbdir(genbuf,currboard);
if((fd = (open(direct, O_RDONLY, 0))) != -1) {
if(lseek(fd, (off_t)(size * (inum1 - 1)), SEEK_SET) !=
-1) {
while(read(fd,&rsfh,size) == size) {
if(i > inum2)
break;
now = getindex(genbuf, rsfh.filename, size);
strcpy(currfile, rsfh.filename);
delete_file(genbuf, sizeof(fileheader_t), now,
cmpfilename);
i++;
}
}
close(fd);
}
}
delete_range(direct, inum1, inum2);
fixkeep(direct, inum1);
if(currmode & MODE_BOARD)
setbtotal(currbid);
return DIRCHANGED;
}
return FULLUPDATE;
}
return DONOTHING;
}
extern char *msg_del_ny;
extern char *msg_del_ok;
static int del_post(int ent, fileheader_t *fhdr, char *direct) {
char genbuf[100];
int not_owned;
boardheader_t *bp;
bp = getbcache(currbid);
if((fhdr->filemode & FILE_MARKED) || (fhdr->filemode & FILE_DIGEST) ||
(fhdr->owner[0] == '-'))
return DONOTHING;
not_owned = strcmp(fhdr->owner, cuser.userid);
if((!(currmode & MODE_BOARD) && not_owned) ||
((bp->brdattr & BRD_VOTEBOARD) && !HAS_PERM(PERM_SYSOP)) ||
!strcmp(cuser.userid, STR_GUEST))
return DONOTHING;
getdata(1, 0, msg_del_ny, genbuf, 3, LCECHO);
if(genbuf[0] == 'y' || genbuf[0] == 'Y') {
strcpy(currfile, fhdr->filename);
setbfile(genbuf,currboard,fhdr->filename);
if(!delete_file (direct, sizeof(fileheader_t), ent, cmpfilename)) {
if(currmode & MODE_SELECT)
{
/* rocker.011018: 利用reference減低loading */
fileheader_t hdr;
int num;
num = fhdr->money & ~FHR_REFERENCE;
setbdir(genbuf, currboard);
get_record(genbuf, &hdr, sizeof (hdr), num);
/* 再這裡要check一下原來的dir裡面是不是有被人動過... */
if (strcmp (hdr.filename, fhdr->filename))
{
num=getindex(genbuf,fhdr->filename,sizeof(fileheader_t));
get_record(genbuf, &hdr, sizeof (hdr), num);
}
/* rocker.011018: 這裡要還原被破壞的money */
fhdr->money = hdr.money;
delete_file (genbuf, sizeof(fileheader_t), num, cmpfilename);
}
cancelpost(fhdr, not_owned);
setbtotal(currbid);
if (fhdr->money < 0)
fhdr->money = 0;
if (not_owned && strcmp(currboard, "Test")){
deumoney(searchuser(fhdr->owner), -fhdr->money);
}
if(!not_owned && strcmp(currboard, "Test")) {
if(cuser.numposts)
cuser.numposts--;
move(b_lines - 1, 0);
clrtoeol();
demoney(-fhdr->money);
passwd_update(usernum, &cuser); /* post 數 */
prints("%s,您的文章減為 %d 篇,支付清潔費 %d 銀", msg_del_ok,
cuser.numposts,fhdr->money);
refresh();
pressanykey();
}
return DIRCHANGED;
}
}
return FULLUPDATE;
}
static int view_postmoney(int ent, fileheader_t *fhdr, char *direct) {
move(b_lines - 1, 0);
clrtoeol();
prints("這一篇文章值 %d 銀", fhdr->money);
refresh();
pressanykey();
return FULLUPDATE;
}
#ifdef OUTJOBSPOOL
/* 看版備份 */
static int tar_addqueue(int ent, fileheader_t *fhdr, char *direct) {
char email[60], qfn[80], ans[2];
FILE *fp;
char bakboard, bakman;
clear();
showtitle("看版備份", BBSNAME);
move(2, 0);
if( !((currmode & MODE_BOARD) || HAS_PERM(PERM_SYSOP)) ) {
move(5, 10);
outs("妳要是版主或是站長才能醬醬啊 -.-\"\"");
pressanykey();
return FULLUPDATE;
}
sprintf(qfn, BBSHOME "/jobspool/tarqueue.%s", currboard);
if( access(qfn, 0) == 0 ){
outs("已經排定行程, 稍後會進行備份");
pressanykey();
return FULLUPDATE;
}
if( !getdata(4, 0, "請輸入目的信箱:", email, sizeof(email), DOECHO) )
return FULLUPDATE;
/* check email -.-"" */
if( strstr(email, "@") == NULL || strstr(email, ".bbs@") != NULL ){
move(6, 0);
outs("您指定的信箱不正確! ");
pressanykey();
return FULLUPDATE;
}
getdata(6, 0, "要備份看版內容嗎(Y/N)?[Y]", ans, sizeof(ans), LCECHO);
bakboard = (ans[0] == 'n' || ans[0] =='N') ? 0 : 1;
getdata(7, 0, "要備份精華區內容嗎(Y/N)?[N]", ans, sizeof(ans), LCECHO);
bakman = (ans[0] == 'y' || ans[0] =='Y') ? 1 : 0;
if( !bakboard && !bakman ){
move(8, 0);
outs("可是我們只能備份看版或精華區的耶 ^^\"\"\"");
pressanykey();
return FULLUPDATE;
}
fp = fopen(qfn, "w");
fprintf(fp, "%s\n", cuser.userid);
fprintf(fp, "%s\n", email);
fprintf(fp, "%d,%d\n", bakboard, bakman);
fclose(fp);
move(10, 0);
outs("系統已經將您的備份排入行程, \n");
outs("稍後將會在系統負荷較低的時候將資料寄給您~ :) ");
pressanykey();
return FULLUPDATE;
}
#endif
static int sequent_ent;
static int continue_flag;
/* ----------------------------------------------------- */
/* 依序讀新文章 */
/* ----------------------------------------------------- */
static int sequent_messages(fileheader_t *fptr) {
static int idc;
char genbuf[200];
if(fptr == NULL)
return (idc = 0);
if(++idc < sequent_ent)
return 0;
if(!brc_unread(fptr->filename,brc_num,brc_list))
return 0;
if(continue_flag)
genbuf[0] = 'y';
else {
prints("讀取文章於:[%s] 作者:[%s]\n標題:[%s]",
currboard, fptr->owner, fptr->title);
getdata(3, 0, "(Y/N/Quit) [Y]: ", genbuf, 3, LCECHO);
}
if(genbuf[0] != 'y' && genbuf[0]) {
clear();
return (genbuf[0] == 'q' ? QUIT : 0);
}
setbfile(genbuf, currboard, fptr->filename);
brc_addlist(fptr->filename);
if(more(genbuf, YEA) == 0)
outmsg("\033[31;47m \033[31m(R)\033[30m回信 \033[31m(↓,n)"
"\033[30m下一封 \033[31m(←,q)\033[30m離開 \033[m");
continue_flag = 0;
switch(egetch()) {
case KEY_LEFT:
case 'e':
case 'q':
case 'Q':
break;
case 'y':
case 'r':
case 'Y':
case 'R':
if(currmode & MODE_POST) {
strcpy(quote_file, genbuf);
do_reply(fptr);
*quote_file = 0;
}
break;
case ' ':
case KEY_DOWN:
case '\n':
case 'n':
continue_flag = 1;
}
clear();
return 0;
}
static int sequential_read(int ent, fileheader_t *fhdr, char *direct) {
char buf[40];
clear();
sequent_messages((fileheader_t *) NULL);
sequent_ent = ent;
continue_flag = 0;
setbdir(buf, currboard);
apply_record(buf, sequent_messages, sizeof(fileheader_t));
return FULLUPDATE;
}
extern char *fn_notes;
extern char *msg_cancel;
extern char *fn_board;
/* ----------------------------------------------------- */
/* 看板備忘錄、文摘、精華區 */
/* ----------------------------------------------------- */
int b_note_edit_bname(int bid) {
char buf[64];
int aborted;
boardheader_t *fh=getbcache(bid);
setbfile(buf, fh->brdname, fn_notes);
aborted = vedit(buf, NA, NULL);
if(aborted == -1) {
clear();
outs(msg_cancel);
pressanykey();
} else {
aborted = (fh->bupdate - now ) / 86400 + 1;
sprintf(buf,"%d", aborted > 0 ? aborted : 0);
getdata_buf(3, 0, "請設定有效期限(0 - 9999)天?", buf, 5, DOECHO);
aborted = atoi(buf);
fh->bupdate = aborted ? now + aborted * 86400 : 0;
substitute_record(fn_board, fh, sizeof(boardheader_t), bid);
}
return 0;
}
static int b_notes_edit() {
if(currmode & MODE_BOARD) {
b_note_edit_bname(currbid);
return FULLUPDATE;
}
return 0;
}
static int b_water_edit() {
if(currmode & MODE_BOARD) {
friend_edit(BOARD_WATER);
return FULLUPDATE;
}
return 0;
}
static int visable_list_edit() {
if(currmode & MODE_BOARD) {
friend_edit(BOARD_VISABLE);
hbflreload(currbid);
return FULLUPDATE;
}
return 0;
}
static int b_post_note() {
char buf[200], yn[3];
if(currmode & MODE_BOARD) {
setbfile(buf, currboard, FN_POST_NOTE );
if(more(buf,NA) == -1) more("etc/"FN_POST_NOTE , NA);
getdata(b_lines - 2, 0, "是否要用自訂post注意事項?", yn, sizeof(yn), LCECHO);
if(yn[0] == 'y')
vedit(buf, NA, NULL);
else
unlink(buf);
return FULLUPDATE;
}
return 0;
}
static int b_application() {
char buf[200];
if(currmode & MODE_BOARD) {
setbfile(buf, currboard, FN_APPLICATION);
vedit(buf, NA, NULL);
return FULLUPDATE;
}
return 0;
}
static int can_vote_edit() {
if(currmode & MODE_BOARD) {
friend_edit(FRIEND_CANVOTE);
return FULLUPDATE;
}
return 0;
}
static int bh_title_edit() {
boardheader_t *bp;
if(currmode & MODE_BOARD) {
char genbuf[BTLEN];
bp = getbcache(currbid);
move(1,0);
clrtoeol();
getdata_str(1,0,"請輸入看板新中文敘述:", genbuf,
BTLEN - 16, DOECHO, bp->title + 7);
if(!genbuf[0])
return 0;
strip_ansi( genbuf,genbuf,0);
strcpy(bp->title + 7,genbuf);
substitute_record(fn_board, bp, sizeof(boardheader_t), currbid);
log_usies("SetBoard", currboard);
return FULLUPDATE;
}
return 0;
}
static int b_notes() {
char buf[64];
setbfile(buf, currboard, fn_notes);
if(more(buf, NA) == -1) {
clear();
move(4, 20);
outs("本看板尚無「備忘錄」。");
}
pressanykey();
return FULLUPDATE;
}
int board_select() {
char fpath[80];
char genbuf[100];
currmode &= ~MODE_SELECT;
sprintf(fpath, "SR.%s", cuser.userid);
setbfile(genbuf, currboard, fpath);
unlink(genbuf);
if(currstat == RMAIL)
sethomedir(currdirect, cuser.userid);
else
setbdir(currdirect, currboard);
return NEWDIRECT;
}
int board_digest() {
if(currmode & MODE_SELECT)
board_select();
currmode ^= MODE_DIGEST;
if(currmode & MODE_DIGEST)
currmode &= ~MODE_POST;
else if (haspostperm(currboard))
currmode |= MODE_POST;
setbdir(currdirect, currboard);
return NEWDIRECT;
}
int board_etc() {
if(!HAS_PERM(PERM_SYSOP))
return DONOTHING;
currmode ^= MODE_ETC;
if(currmode & MODE_ETC)
currmode &= ~MODE_POST;
else if(haspostperm(currboard))
currmode |= MODE_POST;
setbdir(currdirect, currboard);
return NEWDIRECT;
}
extern char *fn_mandex;
static int good_post(int ent, fileheader_t *fhdr, char *direct) {
char genbuf[200];
char genbuf2[200];
int delta = 0;
if((currmode & MODE_DIGEST) || !(currmode & MODE_BOARD))
return DONOTHING;
if(fhdr->filemode & FILE_DIGEST) {
fhdr->filemode = (fhdr->filemode & ~FILE_DIGEST);
if(!strcmp(currboard,"Note") || !strcmp(currboard,"PttBug") ||
!strcmp(currboard,"Artdsn") || !strcmp(currboard, "PttLaw")) {
deumoney(searchuser(fhdr->owner),-1000);
if(!(currmode & MODE_SELECT))
fhdr->money -= 1000;
else
delta = -1000;
}
} else {
fileheader_t digest;
char *ptr, buf[64];
memcpy(&digest, fhdr, sizeof(digest));
digest.filename[0] = 'G';
strcpy(buf, direct);
ptr = strrchr(buf, '/') + 1;
ptr[0] = '\0';
sprintf(genbuf, "%s%s", buf, digest.filename);
if(dashf(genbuf)) unlink (genbuf);
digest.filemode = 0;
sprintf(genbuf2, "%s%s", buf, fhdr->filename);
Link(genbuf2, genbuf);
strcpy(ptr, fn_mandex);
append_record(buf, &digest, sizeof(digest));
fhdr->filemode = (fhdr->filemode & ~FILE_MARKED) | FILE_DIGEST;
if(!strcmp(currboard, "Note") || !strcmp(currboard, "PttBug") ||
!strcmp(currboard,"Artdsn") || !strcmp(currboard, "PttLaw")) {
deumoney(searchuser(fhdr->owner), 1000);
if(!(currmode & MODE_SELECT)) fhdr->money += 1000;
else delta = 1000;
}
}
substitute_record(direct, fhdr, sizeof(*fhdr), ent);
touchdircache(currbid);
/* rocker.011018: 串接模式用reference增進效率 */
if ((currmode & MODE_SELECT) && (fhdr->money & FHR_REFERENCE))
{
fileheader_t hdr;
char genbuf[100];
int num;
num = fhdr->money & ~FHR_REFERENCE;
setbdir(genbuf, currboard);
get_record(genbuf, &hdr, sizeof (hdr), num);
/* 再這裡要check一下原來的dir裡面是不是有被人動過... */
if (strcmp (hdr.filename, fhdr->filename))
{
num = getindex(genbuf, fhdr->filename, sizeof(fileheader_t));
get_record(genbuf, &hdr, sizeof (hdr), num);
}
fhdr->money = hdr.money + delta;
substitute_record(genbuf, fhdr, sizeof(*fhdr), num);
}
return PART_REDRAW;
}
/* help for board reading */
static char *board_help[] = {
"\0全功\能看板操作說明",
"\01基本命令",
"(p)(↑) 上移一篇文章 (^P) 發表文章",
"(n)(↓) 下移一篇文章 (d) 刪除文章",
"(P)(PgUp) 上移一頁 (S) 串連相關文章",
"(N)(PgDn) 下移一頁 (##) 跳到 ## 號文章",
"(r)(→) 閱\讀此篇文章 ($) 跳到最後一篇文章",
"\01進階命令",
"(tab)/z 文摘模式/精華區 (a)(A) 找尋作者",
"(b/f) 展讀備忘錄/參與賭盤 (?)(/) 找尋標題",
"(V/R) 投票/查詢投票結果 (^W)(X) 我在哪裡/推薦文章",
"(x) 轉錄文章到其他看板 (=)/([]<>-+) 找尋首篇文章/主題式閱\讀",
#ifdef INTERNET_EMAIL
"(F) 文章寄回Internet郵箱 (U) 將文章 uuencode 後寄回郵箱",
#endif
"(E) 重編文章 (^H) 列出所有的 New Post(s)",
"\01板主命令",
"(G) 舉辦賭盤/停止下注/開獎(W/w/v) 編輯備忘錄/水桶名單/可看見名單",
"(M/o) 舉行投票/編私投票名單 (m/c/g) 保留文章/選錄精華/文摘",
"(D) 刪除一段範圍的文章 (T/B) 重編文章標題/重編看版標題",
"(i) 編輯申請入會表格 (t/^D) 標記文章/砍除標記的文章",
"(O) 編輯Post注意事項 (H)/(Y) 看板隱藏/現身 取消推薦文章",
NULL
};
static int b_help() {
show_help(board_help);
return FULLUPDATE;
}
/* ----------------------------------------------------- */
/* 板主設定隱形/ 解隱形 */
/* ----------------------------------------------------- */
char board_hidden_status;
#ifdef BMCHS
extern char *fn_board;
static int change_hidden(int ent, fileheader_t *fhdr, char *direct)
{
boardheader_t bh;
int bid;
char ans[4];
if( !((currmode & MODE_BOARD) || HAS_PERM(PERM_SYSOP)) ||
currboard[0] == 0 ||
(bid = getbnum(currboard)) < 0 ||
get_record(fn_board, &bh, sizeof(bh), bid) == -1 )
return DONOTHING;
if( ((bh.brdattr & BRD_HIDE) && (bh.brdattr & BRD_POSTMASK)) ){
getdata(1, 0, "目前板在隱形狀態, 要解隱形嘛(Y/N)?[N]",
ans, sizeof(ans), LCECHO);
if( ans[0] != 'y' && ans[0] != 'Y' )
return FULLUPDATE;
getdata(2, 0, "再確認一次, 真的要把板板公開嘛 @____@(Y/N)?[N]",
ans, sizeof(ans), LCECHO);
if( ans[0] != 'y' && ans[0] != 'Y' )
return FULLUPDATE;
if( bh.brdattr & BRD_HIDE ) bh.brdattr -= BRD_HIDE;
if( bh.brdattr & BRD_POSTMASK ) bh.brdattr -= BRD_POSTMASK;
log_usies("OpenBoard", bh.brdname);
outs("君心今傳眾人,無處不聞弦歌。\n");
board_hidden_status = 0;
hbflreload(bid);
}
else{
getdata(1, 0, "目前板在現形狀態, 要隱形嘛(Y/N)?[N]",
ans, sizeof(ans), LCECHO);
if( ans[0] != 'y' && ans[0] != 'Y' )
return FULLUPDATE;
bh.brdattr |= BRD_HIDE;
bh.brdattr |= BRD_POSTMASK;
log_usies("CloseBoard", bh.brdname);
outs("君心今已掩抑,惟盼善自珍重。\n");
board_hidden_status = 1;
}
setup_man(&bh);
substitute_record(fn_board, &bh, sizeof(bh), bid);
reset_board(bid);
log_usies("SetBoard", bh.brdname);
pressanykey();
return FULLUPDATE;
}
#endif
/* ----------------------------------------------------- */
/* 看板功能表 */
/* ----------------------------------------------------- */
struct onekey_t read_comms[] = {
{KEY_TAB, board_digest},
{'C', board_etc},
{'b', b_notes},
{'c', cite_post},
{'r', read_post},
{'z', b_man},
{'D', del_range},
{'S', sequential_read},
{'E', edit_post},
{'T', edit_title},
{'s', do_select},
{'R', b_results},
{'V', b_vote},
{'M', b_vote_maintain},
{'B', bh_title_edit},
{'W', b_notes_edit},
{'O', b_post_note},
{'w', b_water_edit},
{'v', visable_list_edit},
{'i', b_application},
{'o', can_vote_edit},
{'x', cross_post},
{'X', recommend},
{'Y', recommend_cancel},
{'h', b_help},
#ifndef NO_GAMBLE
{'f', join_gamble},
{'G', hold_gamble},
#endif
{'g', good_post},
{'y', reply_post},
{'d', del_post},
{'m', mark_post},
{'L', solve_post},
{Ctrl('P'), do_post},
{Ctrl('W'), whereami},
{'Q', view_postmoney},
#ifdef OUTJOBSPOOL
{'u', tar_addqueue},
#endif
#ifdef BMCHS
{'H', change_hidden},
#endif
{'\0', NULL}
};
time_t board_visit_time;
int Read() {
int mode0 = currutmp->mode;
int stat0 = currstat, tmpbid=currutmp->brc_id;
char buf[40];
#ifdef LOG_BOARD
time_t usetime = now;
#endif
setutmpmode(READING);
set_board();
if(board_visit_time < board_note_time) {
setbfile(buf, currboard, fn_notes);
more(buf, NA);
pressanykey();
}
setutmpbid(currbid);
setbdir(buf, currboard);
curredit &= ~EDIT_MAIL;
i_read(READING, buf, readtitle, readdoent, read_comms,
currbid);
#ifdef LOG_BOARD
log_board(currboard, now - usetime);
#endif
brc_update();
setutmpbid(tmpbid);
currutmp->mode = mode0;
currstat = stat0;
return 0;
}
void ReadSelect() {
int mode0 = currutmp->mode;
int stat0 = currstat;
char genbuf[200];
currstat = XMODE;
if(do_select(0, 0, genbuf) == NEWDIRECT)
Read();
setutmpbid(0);
currutmp->mode = mode0;
currstat = stat0;
}
#ifdef LOG_BOARD
static void log_board(char *mode, time_t usetime) {
char buf[ 256 ];
if(usetime > 30) {
sprintf(buf, "USE %-20.20s Stay: %5ld (%s) %s",
mode, usetime ,cuser.userid ,ctime(&now));
log_file(FN_USEBOARD,buf);
}
}
#endif
int Select() {
char genbuf[200];
setutmpmode(SELECT);
do_select(0, NULL, genbuf);
return 0;
}