From 2124ed3a627d9f19d51c9b6eb0b6543fb2fd96ed Mon Sep 17 00:00:00 2001 From: piaip Date: Wed, 26 Aug 2009 11:12:56 +0000 Subject: * refine ordersong * merge topsong and ordersong to single ordersong.c * change max song from MAX_MOVIE to MAX_SONG * change random menu 'movie' to sequential slide show mode git-svn-id: http://opensvn.csie.org/pttbbs/trunk/pttbbs@4779 63ad8ddf-47c3-0310-b6dd-a9e9d9715204 --- include/proto.h | 7 +- mbbsd/Makefile | 2 +- mbbsd/cal.c | 176 ----------------------------------- mbbsd/menu.c | 14 ++- mbbsd/ordersong.c | 267 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ mbbsd/topsong.c | 72 --------------- 6 files changed, 284 insertions(+), 254 deletions(-) create mode 100644 mbbsd/ordersong.c delete mode 100644 mbbsd/topsong.c diff --git a/include/proto.h b/include/proto.h index 879d9a8a..80265894 100644 --- a/include/proto.h +++ b/include/proto.h @@ -141,7 +141,6 @@ int give_money_ui(const char *userid); int p_give(void); int p_cloak(void); int p_from(void); -int ordersong(void); int p_exmail(void); int mail_redenvelop(const char* from, const char* to, int money, char *fpath); void resolve_over18(void); @@ -604,9 +603,9 @@ int term_init(void); void term_resize(int w, int h); void bell(void); -/* topsong */ -void sortsong(void); -int topsong(void); +/* ordersong */ +int ordersong(void); +int topsong(void); /* user */ int kill_user(int num, const char *userid); diff --git a/mbbsd/Makefile b/mbbsd/Makefile index 52fb9ce4..6b15e1dc 100644 --- a/mbbsd/Makefile +++ b/mbbsd/Makefile @@ -14,7 +14,7 @@ NETOBJS = mbbsd.o io.o term.o telnet.o TALKOBJS = talk.o chat.o friend.o UTILOBJS = stuff.o kaede.o convert.o name.o syspost.o PAGEROBJS= more.o pmore.o -PLUGOBJS = calendar.o topsong.o gamble.o vice.o angel.o +PLUGOBJS = calendar.o ordersong.o gamble.o vice.o angel.o CHESSOBJS= chess.o chc.o chc_tab.o ch_go.o ch_gomo.o ch_dark.o ch_reversi.o GAMEOBJS = card.o chicken.o OBJS:= admin.o assess.o edit.o menu.o xyz.o var.o visio.o \ diff --git a/mbbsd/cal.c b/mbbsd/cal.c index ec025b90..2e3b441a 100644 --- a/mbbsd/cal.c +++ b/mbbsd/cal.c @@ -81,182 +81,6 @@ vice(int money, const char *item) return 0; } -#define lockreturn(unmode, state) if(lockutmpmode(unmode, state)) return -#define lockreturn0(unmode, state) if(lockutmpmode(unmode, state)) return 0 -#define lockbreak(unmode, state) if(lockutmpmode(unmode, state)) break -#define SONGBOOK "etc/SONGBOOK" -#define OSONGPATH "etc/SONGO" - -static int -osong(void) -{ - char sender[IDLEN + 1], receiver[IDLEN + 1], buf[200], - genbuf[200], filename[256], say[51]; - char trans_buffer[PATHLEN]; - char address[45]; - FILE *fp, *fp1; - //*fp2; - fileheader_t mail; - int nsongs; - char save_title[STRLEN]; - - strlcpy(buf, Cdatedate(&now), sizeof(buf)); - - lockreturn0(OSONG, LOCK_MULTI); - - /* Jaky 一人一天點一首 */ - if (!strcmp(buf, Cdatedate(&cuser.lastsong)) && !HasUserPerm(PERM_SYSOP)) { - move(22, 0); - vmsg("你今天已經點過囉,明天再點吧...."); - unlockutmpmode(); - return 0; - } - - while (1) { - char ans[4]; - move(12, 0); - clrtobot(); - prints("親愛的 %s 歡迎來到歐桑自動點歌系統\n\n", cuser.userid); - outs(ANSI_COLOR(1) "注意點歌內容請勿涉及謾罵 人身攻擊 猥褻" - "公然侮辱 誹謗\n" - "若有上述違規情形,站方將保留決定是否公開播放的權利\n" - "如不同意請按 (3) 離開。" ANSI_RESET "\n"); - getdata(18, 0, "請選擇 " ANSI_COLOR(1) "1)" ANSI_RESET " 開始點歌、" - ANSI_COLOR(1) "2)" ANSI_RESET " 看歌本、" - "或是 " ANSI_COLOR(1) "3)" ANSI_RESET " 離開: ", - ans, sizeof(ans), DOECHO); - - if (ans[0] == '1') - break; - else if (ans[0] == '2') { - a_menu("點歌歌本", SONGBOOK, 0, 0, NULL); - clear(); - } - else if (ans[0] == '3') { - vmsg("謝謝光臨 :)"); - unlockutmpmode(); - return 0; - } - } - - reload_money(); - if (cuser.money < 200) { - move(22, 0); - vmsg("點歌要200元唷!...."); - unlockutmpmode(); - return 0; - } - - getdata_str(19, 0, "點歌者(可匿名): ", sender, sizeof(sender), DOECHO, cuser.userid); - getdata(20, 0, "點給(可匿名): ", receiver, sizeof(receiver), DOECHO); - - getdata_str(21, 0, "想要要對他(她)說..:", say, - sizeof(say), DOECHO, "我愛妳.."); - snprintf(save_title, sizeof(save_title), - "%s:%s", sender, say); - getdata_str(22, 0, "寄到誰的信箱(真實 ID 或 E-mail)?", - address, sizeof(address), LCECHO, receiver); - vmsg("接著要選歌囉..進入歌本好好的選一首歌吧..^o^"); - a_menu("點歌歌本", SONGBOOK, 0, 0, trans_buffer); - if (!trans_buffer[0] || strstr(trans_buffer, "home") || - strstr(trans_buffer, "boards") || !(fp = fopen(trans_buffer, "r"))) { - unlockutmpmode(); - return 0; - } - strlcpy(filename, OSONGPATH, sizeof(filename)); - - stampfile(filename, &mail); - - unlink(filename); - - if (!(fp1 = fopen(filename, "w"))) { - fclose(fp); - unlockutmpmode(); - return 0; - } - strlcpy(mail.owner, "點歌機", sizeof(mail.owner)); - snprintf(mail.title, sizeof(mail.title), "◇ %s 點給 %s ", sender, receiver); - - while (fgets(buf, sizeof(buf), fp)) { - char *po; - if (!strncmp(buf, "標題: ", 6)) { - clear(); - move(10, 10); - outs(buf); - pressanykey(); - fclose(fp); - fclose(fp1); - unlockutmpmode(); - return 0; - } - while ((po = strstr(buf, "<~Src~>"))) { - const char *dot = ""; - if (is_validuserid(sender) && strcmp(sender, cuser.userid) != 0) - dot = "."; - po[0] = 0; - snprintf(genbuf, sizeof(genbuf), "%s%s%s%s", buf, sender, dot, po + 7); - strlcpy(buf, genbuf, sizeof(buf)); - } - while ((po = strstr(buf, "<~Des~>"))) { - po[0] = 0; - snprintf(genbuf, sizeof(genbuf), "%s%s%s", buf, receiver, po + 7); - strlcpy(buf, genbuf, sizeof(buf)); - } - while ((po = strstr(buf, "<~Say~>"))) { - po[0] = 0; - snprintf(genbuf, sizeof(genbuf), "%s%s%s", buf, say, po + 7); - strlcpy(buf, genbuf, sizeof(buf)); - } - fputs(buf, fp1); - } - fclose(fp1); - fclose(fp); - - log_filef("etc/osong.log", LOG_CREAT, "id: %-12s ◇ %s 點給 %s : \"%s\", 轉寄至 %s\n", cuser.userid, sender, receiver, say, address); - - if (append_record(OSONGPATH "/" FN_DIR, &mail, sizeof(mail)) != -1) { - cuser.lastsong = now; - /* Jaky 超過 MAX_MOVIE 首歌就開始砍 */ - nsongs = get_num_records(OSONGPATH "/" FN_DIR, sizeof(mail)); - if (nsongs > MAX_MOVIE) { - // XXX race condition - delete_range(OSONGPATH "/" FN_DIR, 1, nsongs - MAX_MOVIE); - } - snprintf(genbuf, sizeof(genbuf), "%s says \"%s\" to %s.", sender, say, receiver); - log_usies("OSONG", genbuf); - /* 把第一首拿掉 */ - vice(200, "點歌"); - } - snprintf(save_title, sizeof(save_title), "%s:%s", sender, say); - hold_mail(filename, receiver, save_title); - - if (address[0]) { - bsmtp(filename, save_title, address, NULL); - } - clear(); - outs( - "\n\n 恭喜您點歌完成囉..\n" - " 一小時內動態看板會自動重新更新\n" - " 大家就可以看到您點的歌囉\n\n" - " 點歌有任何問題可以到Note板的精華區找答案\n" - " 也可在Note板精華區看到自己的點歌記錄\n" - " 有任何保貴的意見也歡迎到Note板留話\n" - " 讓親切的板主為您服務\n"); - pressanykey(); - sortsong(); - topsong(); - - unlockutmpmode(); - return 1; -} - -int -ordersong(void) -{ - osong(); - return 0; -} - static int inmailbox(int m) { diff --git a/mbbsd/menu.c b/mbbsd/menu.c index bbb540f3..bd598cea 100644 --- a/mbbsd/menu.c +++ b/mbbsd/menu.c @@ -266,7 +266,15 @@ movie(int cmdmode) } else if (cmdmode == 999999) { /* Goodbye my friend */ i = 0; } else { - i = N_SYSMOVIE + (int)(((float)SHM->last_film - N_SYSMOVIE + 1) * (random() / (RAND_MAX + 1.0))); + // do not use random. we work in slide show mode. + // since menu is updated per hour, the total presentation time + // should be less than one hour. 3600/MAX_MOVIE(500)=7. + // syncnow(); + if (SHM->last_film > N_SYSMOVIE) + i = N_SYSMOVIE + (now / (3600 / MAX_MOVIE) ) % + (SHM->last_film+1-N_SYSMOVIE); + else + i = 0; // SHM->last_film; } move(1, 0); @@ -277,6 +285,10 @@ movie(int cmdmode) out_lines(SHM->notes[i], 11, 0); /* 只印11行就好 */ #endif outs(ANSI_RESET); +#ifdef DEBUG + // XXX piaip test + move(FILMROW, 0); prints(" [ %d ] ", i); +#endif } typedef struct { diff --git a/mbbsd/ordersong.c b/mbbsd/ordersong.c new file mode 100644 index 00000000..7163e1a5 --- /dev/null +++ b/mbbsd/ordersong.c @@ -0,0 +1,267 @@ +/* $Id$ */ +#include "bbs.h" + +#define lockreturn(unmode, state) if(lockutmpmode(unmode, state)) return +#define lockreturn0(unmode, state) if(lockutmpmode(unmode, state)) return 0 +#define lockbreak(unmode, state) if(lockutmpmode(unmode, state)) break +#define SONGBOOK "etc/SONGBOOK" +#define OSONGPATH "etc/SONGO" +#define ORDER_SONG_COST (200) // how much to order a song + +#define MAX_SONGS (MAX_MOVIE-100) // (400) XXX MAX_SONGS should be fewer than MAX_MOVIE. + +static void sortsong(void); + +static int +do_order_song(void) +{ + char sender[IDLEN + 1], receiver[IDLEN + 1], buf[200], + genbuf[200], filename[256], say[51]; + char trans_buffer[PATHLEN]; + char address[45]; + FILE *fp, *fp1; + fileheader_t mail; + int nsongs; + char save_title[STRLEN]; + + strlcpy(buf, Cdatedate(&now), sizeof(buf)); + + lockreturn0(OSONG, LOCK_MULTI); + + /* Jaky 一人一天點一首 */ + if (!strcmp(buf, Cdatedate(&cuser.lastsong)) && !HasUserPerm(PERM_SYSOP)) { + move(22, 0); + vmsg("你今天已經點過囉,明天再點吧...."); + unlockutmpmode(); + return 0; + } + + while (1) { + char ans[4]; + move(12, 0); + clrtobot(); + prints("親愛的 %s 歡迎來到自動點歌系統\n\n", cuser.userid); + outs(ANSI_COLOR(1) "注意點歌內容請勿涉及謾罵 人身攻擊 猥褻" + "公然侮辱 誹謗\n" + "若有上述違規情形,站方將保留決定是否公開播放的權利\n" + "如不同意請按 (3) 離開。" ANSI_RESET "\n"); +#ifdef USE_PFTERM + getdata(18, 0, "請選擇 " ANSI_COLOR(1) "1)" ANSI_RESET " 開始點歌、" + ANSI_COLOR(1) "2)" ANSI_RESET " 看歌本、" + "或是 " ANSI_COLOR(1) "3)" ANSI_RESET " 離開: ", + ans, sizeof(ans), DOECHO); +#else + getdata(18, 0, "請選擇 1)開始點歌 2)看歌本 3)離開: ", + ans, sizeof(ans), DOECHO); +#endif + + if (ans[0] == '1') + break; + else if (ans[0] == '2') { + a_menu("點歌歌本", SONGBOOK, 0, 0, NULL); + clear(); + } + else if (ans[0] == '3') { + vmsg("謝謝光臨 :)"); + unlockutmpmode(); + return 0; + } + } + + reload_money(); + if (cuser.money < ORDER_SONG_COST) { + move(22, 0); + vmsgf("點歌要 %d 元唷!....", ORDER_SONG_COST); + unlockutmpmode(); + return 0; + } + + getdata_str(19, 0, "點歌者(可匿名): ", sender, sizeof(sender), DOECHO, cuser.userid); + getdata(20, 0, "點給(可匿名): ", receiver, sizeof(receiver), DOECHO); + + getdata_str(21, 0, "想要要對他(她)說..:", say, + sizeof(say), DOECHO, "我愛妳.."); + snprintf(save_title, sizeof(save_title), + "%s:%s", sender, say); + getdata_str(22, 0, "寄到誰的信箱(真實 ID 或 E-mail)?", + address, sizeof(address), LCECHO, receiver); + vmsg("接著要選歌囉..進入歌本好好的選一首歌吧..^o^"); + a_menu("點歌歌本", SONGBOOK, 0, 0, trans_buffer); + if (!trans_buffer[0] || strstr(trans_buffer, "home") || + strstr(trans_buffer, "boards") || !(fp = fopen(trans_buffer, "r"))) { + unlockutmpmode(); + return 0; + } +#ifdef DEBUG + vmsg(trans_buffer); +#endif + strlcpy(filename, OSONGPATH, sizeof(filename)); + + stampfile(filename, &mail); + + unlink(filename); + + if (!(fp1 = fopen(filename, "w"))) { + fclose(fp); + unlockutmpmode(); + return 0; + } + strlcpy(mail.owner, "點歌機", sizeof(mail.owner)); + snprintf(mail.title, sizeof(mail.title), "◇ %s 點給 %s ", sender, receiver); + + while (fgets(buf, sizeof(buf), fp)) { + char *po; + if (!strncmp(buf, "標題: ", 6)) { + clear(); + move(10, 10); + outs(buf); + pressanykey(); + fclose(fp); + fclose(fp1); + unlockutmpmode(); + return 0; + } + while ((po = strstr(buf, "<~Src~>"))) { + const char *dot = ""; + if (is_validuserid(sender) && strcmp(sender, cuser.userid) != 0) + dot = "."; + po[0] = 0; + snprintf(genbuf, sizeof(genbuf), "%s%s%s%s", buf, sender, dot, po + 7); + strlcpy(buf, genbuf, sizeof(buf)); + } + while ((po = strstr(buf, "<~Des~>"))) { + po[0] = 0; + snprintf(genbuf, sizeof(genbuf), "%s%s%s", buf, receiver, po + 7); + strlcpy(buf, genbuf, sizeof(buf)); + } + while ((po = strstr(buf, "<~Say~>"))) { + po[0] = 0; + snprintf(genbuf, sizeof(genbuf), "%s%s%s", buf, say, po + 7); + strlcpy(buf, genbuf, sizeof(buf)); + } + fputs(buf, fp1); + } + fclose(fp1); + fclose(fp); + + log_filef("etc/osong.log", LOG_CREAT, "id: %-12s ◇ %s 點給 %s : \"%s\", 轉寄至 %s\n", cuser.userid, sender, receiver, say, address); + + if (append_record(OSONGPATH "/" FN_DIR, &mail, sizeof(mail)) != -1) { + cuser.lastsong = now; + /* Jaky 超過 MAX_MOVIE 首歌就開始砍 */ + // XXX 載入的順序會長得像是: + // 3. ◆ <系統> 動態看板 SYSOP [01/23/08] + // 4. ◆ <點歌> 動態看板 Ptt [08/26/09] + // 5. ◆ <廣告> 動態看板 SYSOP [08/22/09] + // 6. ◆ <看板> 動態看板 SYSOP [04/16/09] + // 由於點歌部份算是早載入的,不能直接用 MAX_MOVIE 不然後面都沒得玩。 + nsongs = get_num_records(OSONGPATH "/" FN_DIR, sizeof(mail)); + if (nsongs > MAX_SONGS) { + // XXX race condition + delete_range(OSONGPATH "/" FN_DIR, 1, nsongs - MAX_SONGS); + } + snprintf(genbuf, sizeof(genbuf), "%s says \"%s\" to %s.", + sender, say, receiver); + log_usies("OSONG", genbuf); + vice(ORDER_SONG_COST, "點歌"); + } + snprintf(save_title, sizeof(save_title), "%s:%s", sender, say); + hold_mail(filename, receiver, save_title); + + if (address[0]) { + bsmtp(filename, save_title, address, NULL); + } + clear(); + outs( + "\n\n 恭喜您點歌完成囉...\n" + " 一小時內動態看板會自動重新更新,\n" + " 大家就可以看到您點的歌囉!\n\n" + " 點歌有任何問題可以到 " BN_NOTE " 板的精華區找答案,\n" + " 也可在 " BN_NOTE " 板精華區看到自己的點歌記錄。\n" + " 有任何寶貴的意見也歡迎到 " BN_NOTE " 看板留話,\n" + " 讓親切的板主為您服務。\n"); + pressanykey(); + sortsong(); + topsong(); + + unlockutmpmode(); + return 1; +} + +int +ordersong(void) +{ + do_order_song(); + return 0; +} + +// topsong + +#define QCAST int (*)(const void *, const void *) + +typedef struct songcmp_t { + char name[100]; + char cname[100]; + int count; +} songcmp_t; + + +static int +count_cmp(songcmp_t * b, songcmp_t * a) +{ + return (a->count - b->count); +} + +int +topsong(void) +{ + more(FN_TOPSONG, YEA); + return 0; +} + + +static void +sortsong(void) +{ + FILE *fo, *fp = fopen(BBSHOME "/" FN_USSONG, "r"); + songcmp_t songs[MAX_SONGS + 1]; + int n; + char buf[256], cbuf[256]; + int totalcount = 0; + + memset(songs, 0, sizeof(songs)); + if (!fp) + return; + if (!(fo = fopen(FN_TOPSONG, "w"))) { + fclose(fp); + return; + } + totalcount = 0; + /* XXX: 除了前 MAX_SONGS 首, 剩下不會排序 */ + while (fgets(buf, 200, fp)) { + chomp(buf); + strip_blank(cbuf, buf); + if (!cbuf[0] || !isprint2((int)cbuf[0])) + continue; + + for (n = 0; n < MAX_SONGS && songs[n].name[0]; n++) + if (!strcmp(songs[n].cname, cbuf)) + break; + strlcpy(songs[n].name, buf, sizeof(songs[n].name)); + strlcpy(songs[n].cname, cbuf, sizeof(songs[n].cname)); + songs[n].count++; + totalcount++; + } + qsort(songs, MAX_SONGS, sizeof(songcmp_t), (QCAST) count_cmp); + fprintf(fo, + " " ANSI_COLOR(36) "──" ANSI_COLOR(37) "名次" ANSI_COLOR(36) "──────" ANSI_COLOR(37) + "歌名" ANSI_COLOR(36) "───────────" ANSI_COLOR(37) "次數" ANSI_COLOR(36) + "──" ANSI_COLOR(32) "共%d次" ANSI_COLOR(36) "──" ANSI_RESET "\n", totalcount); + for (n = 0; n < 100 && songs[n].name[0]; n++) { + fprintf(fo, " %5d. %-38.38s %4d " ANSI_COLOR(32) "[%.2f]" ANSI_RESET "\n", n + 1, + songs[n].name, songs[n].count, + (float)songs[n].count / totalcount); + } + fclose(fp); + fclose(fo); +} diff --git a/mbbsd/topsong.c b/mbbsd/topsong.c deleted file mode 100644 index 3ea60dbf..00000000 --- a/mbbsd/topsong.c +++ /dev/null @@ -1,72 +0,0 @@ -/* $Id$ */ -#include "bbs.h" - -#define MAX_SONGS 300 -#define QCAST int (*)(const void *, const void *) - -typedef struct songcmp_t { - char name[100]; - char cname[100]; - int count; -} songcmp_t; - - -static int -count_cmp(songcmp_t * b, songcmp_t * a) -{ - return (a->count - b->count); -} - -int -topsong(void) -{ - more(FN_TOPSONG, YEA); - return 0; -} - - -void -sortsong(void) -{ - FILE *fo, *fp = fopen(BBSHOME "/" FN_USSONG, "r"); - songcmp_t songs[MAX_SONGS + 1]; - int n; - char buf[256], cbuf[256]; - int totalcount = 0; - - memset(songs, 0, sizeof(songs)); - if (!fp) - return; - if (!(fo = fopen(FN_TOPSONG, "w"))) { - fclose(fp); - return; - } - totalcount = 0; - /* XXX: 除了前 MAX_SONGS 首, 剩下不會排序 */ - while (fgets(buf, 200, fp)) { - chomp(buf); - strip_blank(cbuf, buf); - if (!cbuf[0] || !isprint2((int)cbuf[0])) - continue; - - for (n = 0; n < MAX_SONGS && songs[n].name[0]; n++) - if (!strcmp(songs[n].cname, cbuf)) - break; - strlcpy(songs[n].name, buf, sizeof(songs[n].name)); - strlcpy(songs[n].cname, cbuf, sizeof(songs[n].cname)); - songs[n].count++; - totalcount++; - } - qsort(songs, MAX_SONGS, sizeof(songcmp_t), (QCAST) count_cmp); - fprintf(fo, - " " ANSI_COLOR(36) "──" ANSI_COLOR(37) "名次" ANSI_COLOR(36) "──────" ANSI_COLOR(37) "歌" - " 名" ANSI_COLOR(36) "───────────" ANSI_COLOR(37) "次數" ANSI_COLOR(36) "" - "──" ANSI_COLOR(32) "共%d次" ANSI_COLOR(36) "──" ANSI_RESET "\n", totalcount); - for (n = 0; n < 100 && songs[n].name[0]; n++) { - fprintf(fo, " %5d. %-38.38s %4d " ANSI_COLOR(32) "[%.2f]" ANSI_RESET "\n", n + 1, - songs[n].name, songs[n].count, - (float)songs[n].count / totalcount); - } - fclose(fp); - fclose(fo); -} -- cgit v1.2.3