西安 邮 电大学
实验名称
一、实验目的
1.了解计算机内部的字符处理与编码
2. 提高C语言编程能力
二、实验要求
编写一个程序,实现从文件中读取内容,并可以图形界面下显示文章
三、 实验内容
一.函数功能简介:
1.bool read_file(char s[]);
把对应路径文件中的内容读入字符串s中,成功则返回ture ,否则返回false
2.long get_adress(int x,int y,char *s);
遍历一遍s数组,把每行的第一个字符的地址存入adress数组中
3.void print_words(int x,int y,char *s,int colour,intcur);
输出字符:从cur行开始输出,输出PAGE_LINE行
4.DWORD WINAPI mouse(LPVOID l);
多线程函数,在程序运行过程中,一直保持运行状态,不断读取鼠标消息
5.void button_introduce();
作者信息按钮
6.void button_openfile();
打开文件按钮
7.void button();
包含5和6两个按钮
8. BeginBatchDraw();
开始批量绘图。执行后,任何绘图操作都将暂时不输出到屏幕上,直到执行 FlushBatchDraw 或 EndBatchDraw 才将之前的绘图输出。
9, FlushBatchDraw();
这个函数用于执行未完成的绘制任务。
10. EndBatchDraw();
结束批量绘制,并执行未完成的绘制任务。
二.宏简介:
FONTSIZE: 字的放大倍数
WIDE: 每行字占的宽度
STARX: 文章左上角的X坐标
STARY: 文章左上角的Y坐标
SIZE_SCREEMX,SIZE_SCREEMY: 开创图形界面的大小
PAGE_LINE: 每页所显示的行数
三.全局变量介绍:
flag:是否单击了“打开文件”
flag1:是否单击了“作者介绍”
flag3:是否是第一次打开文件
char road[200]; :文件路径
char *adress[1000000];储存每行的第一个字符的地址
四.函数调用关系:
Main:
initgraph(SIZE_SCREEMX,SIZE_SCREEMY);
BeginBatchDraw();
introduce()
button();
button_introduce();
button_openfile();
CreateThread(NULL,0,mouse,NULL,0,NULL);
get_adress(STARX,STARY,s);
print_words(STARX,STARY,s,WHITE,0);
FlushBatchDraw();
EndBatchDraw();
closegraph();
五.主要函数说明
1.void print_words(int x,int y,char *s,int colour,intcur)
{
FILE*fp;
charbuffer[32]; //32字节的字模缓冲区
registeri,j,k; //寄存器变量
unsignedchar qh,wh;
unsignedlong location;
if((fp=fopen("hzk16","rb"))==NULL)//打开文件
{
printf("Can'topen hzk16!");
getch();
exit(0);
}
while(*s!=EOF)//读到文件尾
{
if(s==adress[cur+PAGE_LINE])break;//当显示完本页面,跳出循环
if(*s>0&&*s<160)
{
charch=s[0];
if(ch=='\n')//当遇到换行 ,进行换行操作
{
y+=16*FONTSIZE;
x=STARX;
}
else
{
if(ch=='')//当遇到空格,不用输出,x坐标右移
{
x+=8*FONTSIZE;
}
elseif(ch==9)//同理,遇到制表符,不输出,右移
{
x+=4*8*FONTSIZE;
}
else
{
outtextxy(x,y,ch);//正常ascall码字符输出
x+=8*FONTSIZE;
}
if(x>=WIDE)//当超过宽度,换行
{
y+=16*FONTSIZE;
x=STARX;
}
}
s+=1;//地址右移
}
else
{
if(*s==-1&&*(s+1)==0) //当遇到汉字的空格时,跳过
x+=16*FONTSIZE;
else
{
qh=*s-0xa0; //汉字第一个码数减去160
wh=*(s+1)-0xa0; //汉字第二个码数减去160
location=(94*(qh-1)+(wh-1))*32L;// 计算汉字字模在文件中的位置
fseek(fp,location,SEEK_SET); //在字模中找到相应汉字对应的字模数据位置
fread(buffer,1,32,fp); //从字模文件中读取数据
for(i=0;i<16;i++) //输出:16行,每行两个字节,每个字节8个二进制数,所以为16*16
for(j=0;j<2;j++)
for(k=0;k<8;k++)
if(((buffer[i*2+j]>>(7-k))&0x1)!=NULL)
fontsize(x+8*j+k,y+i,FONTSIZE,colour);
x+=16*FONTSIZE; //位置右移
}
s+=2; //地址右移
if(x>=WIDE) //换行操作
{
y+=16*FONTSIZE;
x=STARX;
}
}
}
}
2.long get_adress(int x,int y,char *s)
//该函数与print_words同理,只是不用输出,只把第一行的地址保存到adress数组中
if(x==STARX) //核心
{
adress[len]=s;
len++;
}
3.DWORD WINAPI mouse(LPVOID l)
{
MOUSEMSGm; // 定义鼠标消息
while(true)//死循环,不断获取
{
//获取一条鼠标消息
m= GetMouseMsg();
switch(m.uMsg)
{
caseWM_LBUTTONDOWN:
if(m.x>=10&&m.x<=80&&m.y>=10&&m.y<=35)
{
flag=1; //全局变量鼠标单击“打开文件”标记
cleardevice();
InputBox(road,100,"请输入文本文件路径:");//弹出输入框
cleardevice();
}
if(m.x>=110&&m.x<=180&&m.y>=10&&m.y<=35)
{
flag1=1;//全局变量鼠标单击“作者结束标记”
introduce();//介绍函数
FlushBatchDraw();
}
break;
}
}
}
来看源码:
#include
#include
#include
#include
#include
#define FONTSIZE 1
#define WIDE 800
#define STARX 70
#define STARY 50
#define SIZE_SCREEMX 1000
#define SIZE_SCREEMY 800
#define PAGE_LINE 40
DWORD WINAPI mouse(LPVOID l);
void button_introduce();
void button_openfile();
void button();
int flag=0,flag1=0,flag3=0;
char road[200];
char *adress[1000000];
bool read_file(char s[]);
void fontsize(int x,int y,int n,int colour);
void print_words(int x,int y,char *s,int colour,int cur);
long get_adress(int x,int y,char *s);
void introduce();
void main()
{
long len,cur=0;
int ch;
char s[100000];
initgraph(SIZE_SCREEMX,SIZE_SCREEMY);
BeginBatchDraw();
CreateThread(NULL,0,mouse,NULL,0,NULL);
loop:if (flag)
{
flag=0;
memset(s,0,sizeof(s));
memset(adress,0,sizeof(adress));
button();
}
else
{
button();
FlushBatchDraw();
if(!flag3) getch();
flag3=1;
}
if(flag1)
{
flag1=0;
cleardevice();
button();
FlushBatchDraw();
}
if(flag||flag1)
{
goto loop;
}
if(!read_file(s))
{
outtextxy(SIZE_SCREEMX/3,SIZE_SCREEMY/3,"该文件不存在!!");
getch();
cleardevice();
}
len=get_adress(STARX,STARY,s);
print_words(STARX,STARY,s,WHITE,0);
FlushBatchDraw();
do
{
ch=getch();
if(flag||flag1)
{
goto loop;
}
switch(ch)
{
case 'w':
if(cur>=1)
{
cur--;
}
cleardevice();
print_words(STARX,STARY,adress[cur],WHITE,cur);
button();
FlushBatchDraw();
break;
case 's':
if(cur1+PAGE_LINE)
{
cur-=PAGE_LINE;
}
cleardevice();
print_words(STARX,STARY,adress[cur],WHITE,cur);
button();
FlushBatchDraw();
break;
case 'd':
if(cur0&&*s<160)
{
char ch=s[0];
if(ch=='\n')//当遇到换行 ,进行换行操作
{
y+=16*FONTSIZE;
x=STARX;
}
else
{
if(ch==' ')//当遇到空格,不用输出,x坐标右移
{
x+=8*FONTSIZE;
}
else if(ch==9)//同理,遇到制表符,不输出,右移
{
x+=4*8*FONTSIZE;
}
else
{
outtextxy(x,y,ch);//正常ascall码字符输出
x+=8*FONTSIZE;
}
if(x>=WIDE)//当超过宽度,换行
{
y+=16*FONTSIZE;
x=STARX;
}
}
s+=1;//地址右移
}
else
{
if(*s==-1&&*(s+1)==0) //当遇到汉字的空格时,跳过
x+=16*FONTSIZE;
else
{
qh=*s-0xa0; //汉字第一个码数减去160
wh=*(s+1)-0xa0; //汉字第二个码数减去160
location=(94*(qh-1)+(wh-1))*32L; // 计算汉字字模在文件中的位置
fseek(fp,location,SEEK_SET); //在字模中找到相应汉字对应的字模数据位置
fread(buffer,1,32,fp); //从字模文件中读取数据
for(i=0;i<16;i++) //输出:16行,每行两个字节,每个字节8个二进制数,所以为16*16
for(j=0;j<2;j++)
for(k=0;k<8;k++)
if(((buffer[i*2+j]>>(7-k))&0x1)!=NULL)
fontsize(x+8*j+k,y+i,FONTSIZE,colour);
x+=16*FONTSIZE; //位置右移
}
s+=2; //地址右移
if(x>=WIDE) //换行操作
{
y+=16*FONTSIZE;
x=STARX;
}
}
}
}
long get_adress(int x,int y,char *s)
//该函数与print_words同理,只是不用输出,只把第一行的地址保存到adress数组中
{
long len=0;
while(*s!=EOF)
{
if(x==STARX) //核心
{
adress[len]=s;
len++;
}
if(*s>0&&*s<160)
{
char ch=*s;
if(ch=='\n')
{
y+=16*FONTSIZE;
x=STARX;
}
else
{
if(ch==' ')
{
x+=8*FONTSIZE;
}
else if(ch==9)
{
x+=4*8*FONTSIZE;
}
else
{
x+=8*FONTSIZE;
}
if(x>=WIDE)
{
y+=16*FONTSIZE;
x=STARX;
}
}
s+=1;
}
else
{
x+=16*FONTSIZE;
s+=2;
if(x>=WIDE)
{
y+=16*FONTSIZE;
x=STARX;
}
}
}
return len;
}
void introduce()
{
int x=SIZE_SCREEMX/3,y=SIZE_SCREEMY/4;
cleardevice();
outtextxy(x,y,"******************************************************");y+=18*FONTSIZE;
outtextxy(x,y,"** 文本显示器 ");y+=18*FONTSIZE;
outtextxy(x,y,"**作者:西安邮电大学卓越班字符处理小组 ");y+=18*FONTSIZE;
outtextxy(x,y,"**功能简介: ");y+=18*FONTSIZE;
outtextxy(x,y,"** 向上移动:w ");y+=18*FONTSIZE;
outtextxy(x,y,"** 向下移动:s ");y+=18*FONTSIZE;
outtextxy(x,y,"** 向上翻页:a ");y+=18*FONTSIZE;
outtextxy(x,y,"** 向下翻页:d ");y+=18*FONTSIZE;
outtextxy(x,y,"** 退出 :q ");y+=18*FONTSIZE;
outtextxy(x,y,"******************************************************");y+=18*FONTSIZE;
}
void button_openfile()
{
setlinecolor(WHITE);
setfillcolor(LIGHTGRAY);
fillrectangle(10,10,80,35);
settextcolor(GREEN);
outtextxy(15,15,"打开文件");
settextcolor(WHITE);
}
void button_introduce()
{
setlinecolor(WHITE);
setfillcolor(LIGHTGRAY);
fillrectangle(110,10,180,35);
settextcolor(LIGHTMAGENTA);
outtextxy(115,15,"作者简介");
settextcolor(WHITE);
}
void button()
{
button_openfile();
button_introduce();
}
void fontsize(int x,int y,int n,int colour)
{
for(int i=0;i=10&&m.x<=80&&m.y>=10&&m.y<=35)
{
flag=1; //全局变量鼠标单击“打开文件”标记
cleardevice();
InputBox(road,100,"请输入文本文件路径:");//弹出输入框
cleardevice();
}
if(m.x>=110&&m.x<=180&&m.y>=10&&m.y<=35)
{
flag1=1;//全局变量鼠标单击“作者结束标记”
introduce();//介绍函数
FlushBatchDraw();
}
break;
}
}
}