《30天自制操作系统》18_day_学习笔记

时间:2024-12-07 21:38:14

harib15a:
  到这里为止,我们已经能实现窗口的切换了。我们发现所有的窗口都有光标闪烁,而我们只希望可以接受输入的窗口有光标闪烁。这里我们先来修改任务A中的光标闪烁,当按下TAB时,如果让A不现实光标,我们让cursor_c为负值。

void HariMain(void)
{ //......
for (;;) {
//.....
if ( <= i && i <= ) {      /* 键盘数据 */
//.....
if (i == + 0x0f) {     /* Tab键 */
if (key_to == ) {     //向CMD任务窗口输入键盘的数据
key_to = ;
make_wtitle8(buf_win, sht_win->bxsize, "task_a", );//A窗口灰色标题
make_wtitle8(buf_cons, sht_cons->bxsize, "console", );//CMD窗口蓝色标题
cursor_c = -;       /* 此时向CMD发送数据,任务A中没有闪烁图标 */
boxfill8(sht_win->buf, sht_win->bxsize, COL8_FFFFFF, cursor_x, , cursor_x + , );
} else {           //向任务A窗口发送数据
key_to = ;
make_wtitle8(buf_win, sht_win->bxsize, "task_a", );//A窗口蓝色标题
make_wtitle8(buf_cons, sht_cons->bxsize, "console", );//CMD窗口灰色
cursor_c = COL8_000000; /* 此时A任务窗口会显示光标闪烁 */
}
//刷新窗口
sheet_refresh(sht_win, , , sht_win->bxsize, );
sheet_refresh(sht_cons, , , sht_cons->bxsize, );
}
//.....
if (cursor_c >= ) {       //这里是显示闪烁图标的代码
//只对A任务窗口进行闪烁的判断
boxfill8(sht_win->buf, sht_win->bxsize, cursor_c, cursor_x, , cursor_x + , );
} //刷线任务A窗口
sheet_refresh(sht_win, cursor_x, , cursor_x + , );
} else if ( <= i && i <= ) { /* 鼠标的数据 */
//......
} else if (i <= ) {          /* 光标定时器 */
if (i != ) {
timer_init(timer, &fifo, );/* 想定时器缓冲区中写入0 */
if (cursor_c >= ) {
cursor_c = COL8_000000;//设置光标背景颜色白色
}
} else {
timer_init(timer, &fifo, );/* 向定时器缓冲区写入1 */
if (cursor_c >= ) { //设置光标背景黑色
cursor_c = COL8_FFFFFF;
}
}
timer_settime(timer, );    //定时时间为count的50次中断计数的时间
if (cursor_c >= ) {      //cursor_c>0表示窗口A为活动窗口,可以显示光标
                //窗口A图层的填充和刷新
boxfill8(sht_win->buf, sht_win->bxsize, cursor_c, cursor_x, , cursor_x + , );
sheet_refresh(sht_win, cursor_x, , cursor_x + , );
}
}
}
}
}

harib15b:
  1、实现CMD窗口的光标的控制,将光标开始闪烁定义为2,停止闪烁定义为3.

//HariMain节选
  if (i == + 0x0f) {         /* 按下的是TAB键 */
    if (key_to == ) {         //这是向CMD任务发送数据
      key_to = ;
      make_wtitle8(buf_win, sht_win->bxsize, "task_a", );
      make_wtitle8(buf_cons, sht_cons->bxsize, "console", );
      cursor_c = -;            /* cursor_c管理的是任务A的光标的闪烁情况,A停止闪烁 */
      boxfill8(sht_win->buf, sht_win->bxsize, COL8_FFFFFF, cursor_x, , cursor_x + , );
      fifo32_put(&task_cons->fifo, ); /* 这个是CMD光标闪烁的情况,此时CMD闪烁打开 */
    } else {               //向任务A窗口发送数据
      key_to = ;
      make_wtitle8(buf_win, sht_win->bxsize, "task_a", );
      make_wtitle8(buf_cons, sht_cons->bxsize, "console", );
      cursor_c = COL8_000000;      /* A开始闪烁 */
      fifo32_put(&task_cons->fifo, ); /* CMD窗口闪烁关闭 */
    }                   //刷新任务A和CMD窗口图层
    sheet_refresh(sht_win, , , sht_win->bxsize, );
    sheet_refresh(sht_cons, , , sht_cons->bxsize, );
  }

  2、对CMD任务窗口进行修改,刚开始时候将cursor_c置为-1,后面的处理和任务A一样

void console_task(struct SHEET *sheet)
{
//i是从CMD缓冲区取得的数据
if (i <= ) {                  /* 光标定时器超时 */
if (i != ) {
timer_init(timer, &task->fifo, ); /* 先向任务缓冲区写入0 */
if (cursor_c >= ) {
cursor_c = COL8_FFFFFF;     //设置黑色
}
} else {
timer_init(timer, &task->fifo, ); /* 想任务缓冲区写入1 */
if (cursor_c >= ) {
cursor_c = COL8_000000;    //设置白色
}
}
timer_settime(timer, );        //定时器时间
}
if (i == ) {                /* CMD光标打开了 */
cursor_c = COL8_FFFFFF;
}
if (i == ) {                /* CMD光标关闭 */
boxfill8(sheet->buf, sheet->bxsize, COL8_000000, cursor_x, , cursor_x + , );
cursor_c = -;              //关闭之后,重新初始化为-1,回到if判断中间
}
if ( <= i && i <= ) {         /* 获得了键盘的数据 */
if (i == + ) {
/* Backspace键 */
if (cursor_x > ) {        //退格之前至少要16个像素的位置,一个字符+光标
                 //退格之后,光标的位置写“ ”
putfonts8_asc_sht(sheet, cursor_x, , COL8_FFFFFF, COL8_000000, " ", );
cursor_x -= ;         //光标左移8个像素
}
} else {
/* 获得的是一般的字符 */
if (cursor_x < ) {       //CMD窗口宽度最大为240个像素
/* 这里处理的是,每输入一个字符,在CMD中写入一个,光标后移8个像素 */
s[] = i - ;
s[] = ;
putfonts8_asc_sht(sheet, cursor_x, , COL8_FFFFFF, COL8_000000, s, );
cursor_x += ;
}
}
}
    /* 重新显示光标 */
if (cursor_c >= ) {             //如果在前面设置了光标的颜色,说明光标可以显示
boxfill8(sheet->buf, sheet->bxsize, cursor_c, cursor_x, , cursor_x + , );
}
sheet_refresh(sheet, cursor_x, , cursor_x + , );//刷新CMD图层
}
}
}

harib15c:
  1、回车键的响应:这里我们要让CMD窗口的回车键进行响应。这是CMD中实现命令的第一步啊!具体做法就是对输入的回车键进行判断,接着进行相关操作。

if (i ==  + 0x1c) {    /* 回车键的键值 */
  if (key_to != ) { /* 表示向CMD窗口发送数据 */
    fifo32_put(&task_cons->fifo, + );//在CMD任务的缓冲区中写入10+256(其他能够区分的数据也行)
  }
}

  2、回车键的操作:接下来我们要修改CMD任务函数console-task我们再其中创建一个cursor_y变量,当按下回车键的时候,cursor_y增加16个像素(一个字符宽8高16像素)

void console_task(struct SHEET *sheet)
{ //......
for (;;) {
//.....
if (i == + ) { /* 从缓冲区取得i的值为10+256.Enter */
if (cursor_y < + ) {
         /* CMD高度最大为28+112个像素 */
         //先用一个空格填充光标的位置
putfonts8_asc_sht(sheet, cursor_x, cursor_y, COL8_FFFFFF, COL8_000000, " ", );
cursor_y += ;//竖直方向向下移动16个像素,(换行)
          /* 先显示第一个字符‘>’ */
putfonts8_asc_sht(sheet, , cursor_y, COL8_FFFFFF, COL8_000000, ">", );
cursor_x = ; //水平方向也向右移动一个字符的位置
}
} //.....
}
}

harib15d:
  上一步我们虽然实现了CMD对回车键的判断和反应,但当到了CMD窗口的最后一行,下面就没有更多的行了(其实窗口最高为140个像素)。下面我们要实现简单的CMD的窗口的滚动,要实现这个,我们只需要将所有的像素向上移动一行(16个像素)然后把最后一行涂黑就行了。

void console_task(struct SHEET *sheet)
{ //.....
for (;;) {
//.....
if (i == + ) {
/* Enter */
/* 用空格将光标擦除 */
putfonts8_asc_sht(sheet, cursor_x, cursor_y, COL8_FFFFFF, COL8_000000, " ", );
if (cursor_y < + ) {         //cursor_y小于140,还没有到最后一行,换行
cursor_y += ;             /* 换行 */
} else {
                      /* 这个时候,cursor_y>140需要进行滚动了 */
for (y = ; y < + ; y++) {  //从第二行开始,都往上移动一行
for (x = ; x < + ; x++) {
                 //把下一行y+16的,重写写到上一行y,x的值相对不变
sheet->buf[x + y * sheet->bxsize] = sheet->buf[x + (y + ) * sheet->bxsize];
}
}
for (y = + ; y < + ; y++) {//把最后一行涂黑
for (x = ; x < + ; x++) {
sheet->buf[x + y * sheet->bxsize] = COL8_000000;
}
}
sheet_refresh(sheet, , , + , + );
}
/* 显示字符 '>' */
putfonts8_asc_sht(sheet, , cursor_y, COL8_FFFFFF, COL8_000000, ">", );
cursor_x = ;
}
//.....
}
}

harib15e:
  mem命令:到这里为止,我们已经对CMD进行了一些准备工作,下面我们来实现第一个命令:mem命令。这个命令用来显示内存的情况。这里我们还把背景图层上显示的内容全部去掉了。

void console_task(struct SHEET *sheet, unsigned int memtotal)
{
char s[], cmdline[];
//.....
for (;;) {
//......
if ( <= i && i <= ) {               /* 键盘的数据 */
if (i == + ) {                 //backspace
if (cursor_x > ) {
                       /* cursor_x光标向后移动8个像素 */
putfonts8_asc_sht(sheet, cursor_x, cursor_y, COL8_FFFFFF, COL8_000000, " ", );
cursor_x -= ;
}
} else if (i == + ) {            //enter
  /* 用空格擦除光标 */
putfonts8_asc_sht(sheet, cursor_x, cursor_y, COL8_FFFFFF, COL8_000000, " ", );
cmdline[cursor_x / - ] = ;       //这里进行命令的判断,必须是cmd
cursor_y = cons_newline(cursor_y, sheet); //回车键之后得到新的cursor_y
  /* 从这里开始执行CMD命令 */
if (cmdline[] == 'm' && cmdline[] == 'e' && cmdline[] == 'm' && cmdline[] == ) {
//下面输出内存信息
sprintf(s, "total %dMB", memtotal / ( * ));
putfonts8_asc_sht(sheet, , cursor_y, COL8_FFFFFF, COL8_000000, s, );
cursor_y = cons_newline(cursor_y, sheet);
sprintf(s, "free %dKB", memman_total(memman) / );
putfonts8_asc_sht(sheet, , cursor_y, COL8_FFFFFF, COL8_000000, s, );
cursor_y = cons_newline(cursor_y, sheet);
cursor_y = cons_newline(cursor_y, sheet);
} else if (cmdline[] != ) {
/* 命令错误的判断 */
putfonts8_asc_sht(sheet, , cursor_y, COL8_FFFFFF, COL8_000000, "Bad command.", );
cursor_y = cons_newline(cursor_y, sheet);
cursor_y = cons_newline(cursor_y, sheet);
}
//.....
}
}
}

harib15f:
  CLS命令:清空屏幕,在linux中是clear.这里我们使用C语言的字符串比较函strcmmp(#include <string.h>)函数来进行输入命令的判断

void console_task(struct SHEET *sheet, unsigned int memtotal)
{ //.....
for (;;) {
//.....
if ( <= i && i <= ) {           /* 键盘数据(通过任务A) */
//.....
} else if (strcmp(cmdline, "cls") == ) {
                    /* cls命令 */
for (y = ; y < + ; y++) { //这里做的事就是把整个窗口涂成黑色的
for (x = ; x < + ; x++) {
sheet->buf[x + y * sheet->bxsize] = COL8_000000;
}
}
sheet_refresh(sheet, , , + , + );
cursor_y = ;
//.....
}
}