3 MML播放器(harib27c)
我们来想想看,用现存的API,还能做出什么有趣的应用程序呢?想起来了,做个演奏音乐的程序吧,这样一来我们就可以边听音乐边看电子书了,好不惬意。不过说是音乐,其实只是用音质很烂的蜂鸣器来演奏啦。
先来看看实际运行的画面。输入“mmlplay daigo.mml”后如下图。
正在演奏daigo.mml1
1 乐曲名称为《c小调第五交响曲“命运”op.67》。——译者注
■■■■■
在命令行中指定的文件名,是一个用MML(music macro language)编写的文本文件。这个文件中包含的内容是乐谱数据,如“哆来咪发唆啦西哆”分别对应“CDEFGABC”(在C大调中)。在音符的后面加上“+”或者“#”表示升高半音,加上“-”表示降低半音。再后面还可以加上数字,“4”表示四分音符,“8”表示八分音符,“1”表示全音符,如果是“2.”的话则表示符点二分音符。
“O”命令用来设定八度音区,例如可以设为“O4”。“L”命令用来设定音长,表示音符或者休止符(R)在不指定音长时的默认音长。“T”命令用来设定乐曲速度,“>”用来升高一个八度,“<”用来降低一个八度,“&”表示连音线。“Q”用来指定音符演奏是短促还是连贯,如“Q4”标出断奏,“Q8”表示一直使用连音奏法。
“$”开头的是扩展命令,是笔者自己对MML语法扩展出来的。“$K”是卡拉OK命令,后面指定的字符串会显示在卡拉OK栏中。“$E”表示卡拉OK歌词数据的字符编码,不过这个信息在mmlplay.hrb中是被忽略的。
MML数据中是不区分大小写的,演奏到最后会自动回到开头重新演奏,按下“Q”可以退出。
在演奏过程中有一条蓝色的竖线在不停地动,它是根据音阶来移动的,笔者觉得这个看上去挺好玩的。
乐曲数据我们先准备了4个2,选曲按照笔者的喜好(笑)。乐曲数据文件貌似都比较大,因此全部用tek5进行了压缩。
2 这里的几首歌曲给出的是日文版的歌词,如果大家没有对日文显示支持的部分进行改造,而是沿用作者编写的日文显示程序和日文字库,那么在这里只有用日文的歌词才能正常显示。如果大家对日文显示的部分进行了改造,并使用了中文字库,那么就需要将歌词替换为中文。为了方便大家使用,我们在原文下面给出了相应的中文歌词。——译者注
■■■■■
kirakira.mml
- /* 《小星星》3 法国民歌 */
- $E"SJIS"; T120L4O4
- $K"一闪一闪亮晶晶,满天都是小星星"; CCGGAAG2 FFEEDDC2
- $K"挂在天上放光明,好像许多小眼睛"; GGFFEED2 GGFFEED2
- $K"一闪一闪亮晶晶,满天都是小星星"; CCGGAAG2 FFEEDDC2
- $K""; R1
- $K"一闪一闪亮晶晶,满天都是小星星"; CCGGAAG2 FFEEDDC2
- $K"挂在天上放光明,好像许多小眼睛"; GGFFEED2 GGFFEED2
- $K"一闪一闪亮晶晶,满天都是小星星"; CCGGAAG2 FFEEDDC2
- $K""; R1 R1
3 这首歌中文版歌词只有一段,第二段重复第一段歌词即可。——译者注
fujisan.mml
- /* 《富士山》4日本文部省民歌*/
- $E"SJIS"; T120L4O4
- $K"冲破云霄"; G.G8AGEC8D8E2 D.G8GF8E8D2.R
- $K"傲视群山"; G.G8ECA.B8>C<A G.A8G8F8E8D8C2.R
- $K"问鼎惊雷"; D.D8DDC8D8E8F8G2 A.B8>C<AG2.R
- $K"富士,日本第一山"; >C2<AGE.E8AG FED.C8C2.R
- $K"高耸蓝天"; G.G8AGEC8D8E2 D.G8GF8E8D2.R
- $K"银装素裹"; G.G8ECA.B8>C<A G.A8G8F8E8D8C2.R
- $K"身披彩霞"; D.D8DDC8D8E8F8G2 A.B8>C<AG2.R
- $K"富士,日本第一山"; >C2<AGE.E8AG FED.C8C2.R
- $K""; R1R1
4 这首歌是日本的歌谣,没有相应的中文版歌词,这里给出翻译的歌词大意。——译者注
daigo.mml
- /* "第五交响曲 c小调 "命运" op. 67"选段 路德维希•范•贝多芬*/
- $E"SJIS"; $K"第五交响曲 c小调 "命运" op.67";
- T155Q7L8O4
- RGGGE-2.RFFFD4&D1
- RGGGQ8E-Q7A-A-A-Q8GQ7>E-E-E-C8&C2<GGG
- Q8DQ7A-A-A-Q8GQ7>FFFD8&D2GGF
- Q8E-Q7<E-E-FQ8G>Q7GGFQ8E-Q7<E-E-F
- Q8GQ7>GGFL4E-RCRG2.L8R<A-A-A-F4&F1
- RA-A-A-Q8FQ7DDDQ8<BQ7A-A-A-Q8GQ7<GGG
- >>E-A-A-A-Q8FQ7DDDQ8<BQ7A-A-A-Q8GQ7<GGG
- >>E-G>CCQ8C2Q7<BBB>DQ8D2Q7CCCE-Q8E-Q7DQ4DF
- Q8FQ7EQ4EGQ7GQ7FQ4FA-Q8A-Q7GQ4GB-Q8B-Q7A-Q4A->C
- Q8CQ7<BQ4B>DQ7CE-E-E-C<GGGE-C<GGE-CCC<B>>>FDD
- <BGFFD<BGFD<B>CCC>>E-E-E-
- C<AAAG-E-E-E-C<AAAA4R2R4B-4R4
- RB-B-B-E-2F2Q8<B-2>L4B->E-DE-FQ7CQ8CQ7<B-
- Q8B->E-DE-FQ7CQ8CQ7<B-Q8>B->E-DE-FQ7CQ8C<B-
- Q8<B->CD-Q7CQ8<B->C<B-Q7A-Q8>D-E-FQ7E-
- Q8D-E-D-Q7CQ8E-FG-FE-Q7FQ8G-FE-Q7FQ8G-F
- E-Q7FQ8G-FE-FG-FG-Q7AL8B-&B-2Q4>C<B-A-
- Q8A-Q7GQ4FE-Q8E-Q7DQ4CDQ8FQ7E-Q4<B-G
- Q8>DQ7CQ4<A-FQ8>CQ7<B-Q4GE-Q7<B-Q8>>AB-A
- B-AB-Q7AQ4B->C<B-A-Q8A-Q7GQ4FE-Q8E-Q7DQ4CD
- Q8FQ7E-Q4<B-GQ8>DQ7CQ4<A-FQ8>CQ7<B-Q4GE-
- Q7<B->B->B-B-E-GGGE-<B-B-B-GE-E-E-
- Q8<B-Q7>DDDQ8E-Q7>GGGE-<B-B-B-GE-E-E-
- Q8<B-Q7>B-B-B-B-4R4.B-B-B-B-4R4.>DDDE-4
- R1R4
daiku.mml
- /* "第九交响曲 d小调 "合唱" op.125"选段 路德维希•范•贝多芬5 */
- $E"SJIS"; T110L4
- O4
- $K"欢乐女神,圣洁美丽,灿烂光芒照大地"; F+F+GA AGF+E DDEF+ F+.E8E2
- $K"我们心中充满热情,来到你的圣殿里"; F+F+GA AGF+E DDEF+ E.D8D2
- $K"你的力量能使人们消除一切分歧"; EEF+D EF+8G8F+D EF+8G8F+E DE<A>
- $K"在你光辉照耀下面,人们团结成兄弟"; F+& F+F+GA AGF+E DDEF+ E.D8D2
- O5
- $K"在这美丽大地上,普世众生共欢乐"; F+F+GA AGF+E DDEF+ F+.E8E2
- $K"一切人们不论善恶,都蒙自然赐恩泽"; F+F+GA AGF+E DDEF+ E.D8D2
- $K"它给我们爱情美酒,同生共死好朋友"; EEF+D EF+8G8F+D EF+8G8F+E DE<A>
- $K"它让众生共享欢乐,天使也高声同唱歌"; F+& F+F+GA AGF+E DDEF+ E.D8D2
- $K""; R1
5 《欢乐颂》中文歌词共有4段,这里选了有代表性的第1段和第3段。——译者注
■■■■■
然后我们来看程序。
mmlplay.c
#include "apilib.h"
#include <string.h> /* strlen */
int strtol(char *s, char **endp, int base); /*标准函数(stdlib.h) */
void waittimer(int timer, int time);
void end(char *s);
void HariMain(void)
{
char winbuf[256 * 112], txtbuf[100 * 1024];
char s[32], *p, *r;
int win, timer, i, j, t = 120, l = 192 / 4, o = 4, q = 7, note_old = 0;
/*音号与频率(mHz)的对照表*/
/*例如,04A为440Hz,即440000 */
/*第16八度的A为1802240Hz,即1802240000 */
/*以下为第16八度的列表(C~B) */
static int tonetable[12] = {
1071618315, 1135340056, 1202850889, 1274376125, 1350154473, 1430438836,
1515497155, 1605613306, 1701088041, 1802240000, 1909406767, 2022946002
};
static int notetable[7] = { +9, +11, +0 /* C */, +2, +4, +5, +7 };
/*命令行解析*/
api_cmdline(s, 30);
for (p = s; *p > ' '; p++) { } /*一直读到空格为止*/
for (; *p == ' '; p++) { } /*跳过空格*/
i = strlen(p);
if (i > 12) {
file_error:
end("file open error.\n");
}
if (i == 0) {
end(0);
}
/*准备窗口*/
win = api_openwin(winbuf, 256, 112, -1, "mmlplay");
api_putstrwin(win, 128, 32, 0, i, p);
api_boxfilwin(win, 8, 60, 247, 76, 7);
api_boxfilwin(win, 6, 86, 249, 105, 7);
/*载入文件*/
i = api_fopen(p);
if (i == 0) {
goto file_error;
}
j = api_fsize(i, 0);
if (j >= 100 * 1024) {
j = 100 * 1024 - 1;
}
api_fread(txtbuf, j, i);
api_fclose(i);
txtbuf[j] = 0;
r = txtbuf;
i = 0; /*通常模式*/
for (p = txtbuf; *p != 0; p++) { /*为了方便处理,将注释和空白删去*/
if (i == 0 && *p > ' ') { /*不是空格或换行符*/
if (*p == '/') {
if (p[1] == '*') {
i = 1;
} else if (p[1] == '/') {
i = 2;
} else {
*r = *p;
if ('a' <= *p && *p <= 'z') {
*r += 'A' - 'a'; /*将小写字母转换为大写字母*/
}
r++;
}
} else if (*p == 0x22) {
*r = *p;
r++;
i = 3;
} else {
*r = *p;
r++;
}
} else if (i == 1 && *p == '*' && p[1] == '/') { /*段注释*/
p++;
i = 0;
} else if (i == 2 && *p == 0x0a) { /*行注释*/
i = 0;
} else if (i == 3) { /*字符串*/
*r = *p;
r++;
if (*p == 0x22) {
i = 0;
} else if (*p == '%') {
p++;
*r = *p;
r++;
}
}
}
*r = 0;
/*定时器准备*/
timer = api_alloctimer();
api_inittimer(timer, 128);
/*主体*/
p = txtbuf;
for (;;) {
if (('A' <= *p && *p <= 'G') || *p == 'R') { /*音符、休止符*/
/*计算频率*/
if (*p == 'R') {
i = 0;
s[0] = 0;
} else {
i = o * 12 + notetable[*p - 'A'] + 12;
s[0] = 'O';
s[1] = '0' + o;
s[2] = *p;
s[3] = ' ';
s[4] = 0;
}
p++;
if (*p == '+' || *p == '-' || *p == '#') {
s[3] = *p;
if (*p == '-') {
i--;
} else {
i++;
}
p++;
}
if (i != note_old) {
api_boxfilwin(win + 1, 32, 36, 63, 51, 8);
if (s[0] != 0) {
api_putstrwin(win + 1, 32, 36, 10, 4, s);
}
api_refreshwin(win, 32, 36, 64, 52);
if (28 <= note_old && note_old <= 107) {
api_boxfilwin(win, (note_old - 28) * 3 + 8, 60, (note_old - 28) * 3 + 10, 76, 7);
}
if (28 <= i && i <= 107) {
api_boxfilwin(win, (i - 28) * 3 + 8, 60, (i - 28) * 3 + 10, 76, 4);
}
if (s[0] != 0) {
api_beep(tonetable[i % 12] >> (17 - i / 12));
} else {
api_beep(0);
}
note_old = i;
}
/*音长计算*/
if ('0' <= *p && *p <= '9') {
i = 192 / strtol(p, &p, 10);
} else {
i = l;
}
for (; *p == '.'; ) {
p++;
i += i / 2;
}
i *= (60 * 100 / 48);
i /= t;
if (s[0] != 0 && q < 8 && *p != '&') {
j = i * q / 8;
waittimer(timer, j);
api_boxfilwin(win, 32, 36, 63, 51, 8);
if (28 <= note_old && note_old <= 107) {
api_boxfilwin(win, (note_old - 28) * 3 + 8, 60, (note_old - 28) * 3 + 10, 76, 7);
}
note_old = 0;
api_beep(0);
} else {
j = 0;
if (*p == '&') {
p++;
}
}
waittimer(timer, i - j);
} else if (*p == '<') { /*八度-- */
p++;
o--;
} else if (*p == '>') { /*八度++ */
p++;
o++;
} else if (*p == 'O') { /*八度指定*/
o = strtol(p + 1, &p, 10);
} else if (*p == 'Q') { /* Q参数指定*/
q = strtol(p + 1, &p, 10);
} else if (*p == 'L') { /*默认音长指定*/
l = strtol(p + 1, &p, 10);
if (l == 0) {
goto syntax_error;
}
l = 192 / l;
for (; *p == '.'; ) {
p++;
l += l / 2;
}
} else if (*p == 'T') { /*速度指定*/
t = strtol(p + 1, &p, 10);
} else if (*p == '$') { /*扩展命令*/
if (p[1] == 'K') { /*卡拉OK命令*/
p += 2;
for (; *p != 0x22; p++) {
if (*p == 0) {
goto syntax_error;
}
}
p++;
for (i = 0; i < 32; i++) {
if (*p == 0) {
goto syntax_error;
}
if (*p == 0x22) {
break;
}
if (*p == '%') {
s[i] = p[1];
p += 2;
} else {
s[i] = *p;
p++;
}
}
if (i > 30) {
end("karaoke too long.\n");
}
api_boxfilwin(win + 1, 8, 88, 247, 103, 7);
s[i] = 0;
if (i != 0) {
api_putstrwin(win + 1, 128 - i * 4, 88, 0, i, s);
}
api_refreshwin(win, 8, 88, 248, 104);
}
for (; *p != ';'; p++) {
if (*p == 0) {
goto syntax_error;
}
}
p++;
} else if (*p == 0) {
p = txtbuf;
} else {
syntax_error:
end("mml syntax error.\n");
}
}
}
void waittimer(int timer, int time)
{
int i;
api_settimer(timer, time);
for (;;) {
i = api_getkey(1);
if (i == 'Q' || i == 'q') {
api_beep(0);
api_end();
}
if (i == 128) {
return;
}
}
}
void end(char *s)
{
if (s != 0) {
api_putstr0(s);
}
api_beep(0);
api_end();
}
只要读过之前的程序,这段程序应该不会难懂吧。由音符数据计算频率,以及由音符的长度计算定时器应该设定为几秒这些地方可能不太容易理解,其实只要有相应的音乐知识(例如一个音升高半音后,频率是原来的约1.059463倍)就很容易看懂了。
总之(老生常谈了),本书的目的是编写操作系统,而不是编写MML播放器,关于相关音乐理论的讲解就省略了,不好意思。当然,笔者也是想给大家仔细讲讲的,只不过要讲清楚的话,又得增加1章的篇幅了。
将这个程序make一下,得到的mmlplay.hrb大小为1975字节,还是非常小的,真不错!