您的位置 首页 编程

s3c2440的LCD字符显现

在上一篇文章中,我们详细介绍了如何驱动LCD,对于在LCD屏上呈现各种简单的图形已经不是一件高不可攀的难事。但如何绘制字符呢?其实每一字…

在上一篇文章中,咱们详细介绍了怎么驱动LCD,关于在LCD屏上出现各种简略的图形现已不是一件高不可攀的难事。但怎么制作字符呢?

其实每一字符便是一幅图画,字符的巨细对应于图画的巨细,字符的笔画对应于图画的内容。那么怎么把字符转换为图画呢?简略的办法是运用“字模提取”之类的软件,它能够把恣意的字符转换为一个字节型的数组,数组元素中的每一位代表LCD上的一个像素点,当为1时,一共该方位为字符的一个笔画,需求上色,而为0时,一共不是笔画,不需求上色。例如,一个字符想要在16×16的面积上显现,即该字符的宽和高各为16个像素,由于每一个像素用一位来一共,因而用字模提取软件生成的字节型数组,一共有16×16÷8=32个字节。在字模提取的过程中,还要留意取模的次序,次序不同,得到的数组就不同,一般来说是从字符的左上角开端,从左向右,从上到下取模,这样程序编写上会便利一些。相同字体巨细的中文字符和ASCII码字符的宽度还有所不同,一般ASCII码字符的宽度是中文字符宽度的一半,所以显现中文字符的程序和显现ASCII码字符的程序还略有不同。

当把一个字符取模变成一个数组后,只需对该数组中每个元素的每一位顺次进行判别,对值为1的位和值为0的位进行不同的上色处理,即可完结一个字符的制作。

下面的程序给出了一个简略的显现中文字符和ASCII字符的比如,我是用PCtoLCD这款软件来提取程序中想要显现的字符的。

#define U32 unsigned int
#define M5D(n)((n) & 0x1fffff)// To get lower 21bits

#define rGPCCON(*(volatile unsigned *)0x56000020)//Port C control
#define rGPCDAT(*(volatile unsigned *)0x56000024)//Port C data
#define rGPCUP(*(volatile unsigned *)0x56000028)//Pull-up control C

#define rGPDCON(*(volatile unsigned *)0x56000030)//Port D control
#define rGPDDAT(*(volatile unsigned *)0x56000034)//Port D data
#define rGPDUP(*(volatile unsigned *)0x56000038)//Pull-up control D

#define rGPGCON(*(volatile unsigned *)0x56000060)//Port G control
#define rGPGDAT(*(volatile unsigned *)0x56000064)//Port G data
#define rGPGUP(*(volatile unsigned *)0x56000068)//Pull-up control G

#define rLCDCON1(*(volatile unsigned *)0x4d000000)//LCD control 1
#define rLCDCON2(*(volatile unsigned *)0x4d000004)//LCD control 2
#define rLCDCON3(*(volatile unsigned *)0x4d000008)//LCD control 3
#define rLCDCON4(*(volatile unsigned *)0x4d00000c)//LCD control 4
#define rLCDCON5(*(volatile unsigned *)0x4d000010)//LCD control 5
#define rLCDSADDR1(*(volatile unsigned *)0x4d000014)//STN/TFT Frame buffer start address 1
#define rLCDSADDR2(*(volatile unsigned *)0x4d000018)//STN/TFT Frame buffer start address 2
#define rLCDSADDR3(*(volatile unsigned *)0x4d00001c)//STN/TFT Virtual screen address set
#define rLCDINTMSK(*(volatile unsigned *)0x4d00005c)//LCD Interrupt mask
#define rTCONSEL(*(volatile unsigned *)0x4d000060)//LPC3600 Control — edited by junon

#define LCD_WIDTH320
#define LCD_HEIGHT240

#define VSPW(3-1)
#define VBPD(15-1)
#define VFPD(12-1)

#define HSPW(30-1)
#define HBPD(38-1)
#define HFPD(20-1)

#define LINEVAL(LCD_HEIGHT-1)
#define HOZVAL(LCD_WIDTH-1)

//for LCDCON1
#define CLKVAL_TFT6
#define MVAL_USED0
#define PNRMODE_TFT3
#define BPPMODE_TFT13
//#define VIDEO_OUT0

//for LCDCON5
#define BPP24BL0
#define INVVCLK0
#define INVVLINE1
#define INVVFRAME1
#define INVVD0
#define INVVDEN0
#define PWREN1
#define BSWP0
#define HWSWP0

volatile U32 LCD_BUFFER[LCD_HEIGHT][LCD_WIDTH];

unsigned char zhao[]=//赵
{
0x08, 0x00, 0x08, 0x00, 0x08, 0x04, 0x7E, 0x84, 0x08, 0x48, 0x08, 0x28, 0xFF, 0x10, 0x08, 0x10,
0x28, 0x28, 0x2F, 0x28, 0x28, 0x44, 0x28, 0x84, 0x58, 0x00, 0x48, 0x00, 0x87, 0xFE, 0x00, 0x00
};
unsigned char chun[]=//春
{
0x01, 0x00, 0x01, 0x00, 0x3F, 0xFC, 0x01, 0x00, 0x1F, 0xF8, 0x02, 0x00, 0xFF, 0xFE, 0x04, 0x20,
0x08, 0x18, 0x3F, 0xEE, 0xC8, 0x24, 0x0F, 0xE0, 0x08, 0x20, 0x08, 0x20, 0x0F, 0xE0, 0x00, 0x00
};
unsigned char jiang[]=//江
{
0x20, 0x00, 0x10, 0x00, 0x13, 0xFC, 0x00, 0x40, 0x88, 0x40, 0x48, 0x40, 0x50, 0x40, 0x10, 0x40,
0x10, 0x40, 0x20, 0x40, 0xE0, 0x40, 0x20, 0x40, 0x20, 0x40, 0x2F, 0xFE, 0x20, 0x00, 0x00, 0x00,
};
unsigned char ASCII_A[]=//A
{
0x00, 0x00, 0x00, 0x10, 010, 0x18, 0x28, 0x28, 0x24, 0x3C, 0x44, 0x42, 0x42, 0xE7, 0x00, 0x00
};
unsigned char ASCII_R[]=//R
{
0x00, 0x00, 0x00, 0xFC, 0x42, 0x42, 0x42, 0x7C, 0x48, 0x48, 0x44, 0x44, 0x42, 0xE3, 0x00, 0x00
};
unsigned char ASCII_M[]=//M
{
0x00, 0x00, 0x00, 0xEE, 0x6C, 0x6C, 0x6C, 0x6C, 0x54, 0x54, 0x54, 0x54, 0x54, 0xD6, 0x00, 0x00
};

//制作布景
void Brush_Background( U32 c)
{
int x,y ;

for( y = 0 ; y < LCD_HEIGHT ; y++ )
{
for( x = 0 ; x < LCD_WIDTH ; x++ )
{
LCD_BUFFER[y][x] = c ;
}
}
}

//制作像素点
void PutPixel(U32 x,U32 y, U32 c )
{
LCD_BUFFER[y][x] = c;
}

//制作巨细为16×16的中文字符
void Draw_Text16(U32 x,U32 y,U32 color,const unsigned char ch[])
{
unsigned short int i,j;
unsigned char mask,buffer;

for(i=0;i<16;i++)
{
mask=0x80;//掩码
buffer=ch[i*2];//提取一行的第一个字节
for(j=0;j<8;j++)
{
if(buffer&mask)
{
PutPixel(x+j,y+i,color);//为笔画上色
}
mask=mask>>1;
}
mask=0x80;//掩码
buffer=ch[i*2+1];//提取一行的第二个字节
for(j=0;j<8;j++)
{
if(buffer&mask)
{
PutPixel(x+j+8,y+i,color);//为笔画上色
}
mask=mask>>1;
}
}
}

//制作巨细为8×16的ASCII码
void Draw_ASCII(U32 x,U32 y,U32 color,const unsigned char ch[])
{
unsigned short int i,j;
unsigned char mask,buffer;

for(i=0;i<16;i++)
{
mask=0x80;
buffer=ch[i];
for(j=0;j<8;j++)
{
if(buffer&mask)
{
PutPixel(x+j,y+i,color);
}
mask=mask>>1;
}
}
}

//LCD初始化
void LCD_Init()
{
rGPCUP= 0x00000000;
rGPCCON = 0xaaaa02a9;

rGPDUP= 0x00000000;
rGPDCON=0xaaaaaaaa; //Initialize VD[15:8]
rLCDCON1=(CLKVAL_TFT<<8)|(MVAL_USED<<7)|(PNRMODE_TFT<<5)|(BPPMODE_TFT<<1)|0;
rLCDCON2=(VBPD<<24)|(LINEVAL<<14)|(VFPD<<6)|(VSPW);
rLCDCON3=(HBPD<<19)|(HOZVAL<<8)|(HFPD);
rLCDCON4=(HSPW);
rLCDCON5 = (BPP24BL<<12) | (INVVCLK<<10) | (INVVLINE<<9) | (INVVFRAME<<8) | (0<<7) | (INVVDEN<<6) | (PWREN<<3)|(BSWP<<1) | (HWSWP); rLCDSADDR1=(((U32)LCD_BUFFER>>22)<<21)|M5D((U32)LCD_BUFFER>>1);
rLCDSADDR2=M5D( ((U32)LCD_BUFFER+(LCD_WIDTH*LCD_HEIGHT*4))>>1 );
rLCDSADDR3=LCD_WIDTH*4/2;
rLCDINTMSK|=(3); // MASK LCD Sub Interrupt
rTCONSEL = 0;// Disable LPC3480

rGPGUP=rGPGUP&(~(1<<4))|(1<<4); // Pull-up disable
rGPGCON=rGPGCON&(~(3<<8))|(3<<8); //GPG4=LCD_PWREN
rGPGDAT = rGPGDAT | (1<<4) ;
rLCDCON5=rLCDCON5&(~(1<<3))|(1<<3);// PWREN
rLCDCON5=rLCDCON5&(~(1<<5))|(0<<5);// INVPWREN
}

void Main(void)
{

LCD_Init();

rLCDCON1|=1;//舱位LCD显现

Brush_Background(0xFFFFFF);//制作白色布景

//制作黑色字符
Draw_Text16(50,100,0×0,zhao);
Draw_Text16(66,100,0×0,chun);
Draw_Text16(82,100,0×0,jiang);
Draw_ASCII(50,120,0×0,ASCII_A);
Draw_ASCII(58,120,0×0,ASCII_R);
Draw_ASCII(66,120,0×0,ASCII_M);
while(1)
{
;
}
}

看了上面的程序,有人可能会问,假如要在程序中显现很多的中文字符,是不是要把这些字符都取模啊?答复是必定的,但前人现已为咱们完结了这一步,做成了数据库,而且进行了编码,只需依照编码规矩调用该库文件,就能够检索到相要的字符。下面就来说说编码规矩:每个汉字是由两个字节一共的,前一个字节一共的区号,后一个字节一共的位号,那么汉字在汉字库中的方位为:94×(区号-1)+(位号-1)。94一共的是每个区里一共有94个汉字,减1一共的是数组是从0开端,而区号和位号是从1开端的。详细到汉字在某一数据库中的方位,还需求乘以一个汉字字模所占的字节数,即[94×(区号-1)+(位号-1)]×一个汉字字模所占字节数。如一个字模巨细为16×16的宋体数据库,库里每个汉字所占的字节为16×16÷8=32,则每个汉字在该宋体数据库中的方位为:[94×(区号-1)+(位号-1)]×32。ASCII码的字符调用比汉字字符要简略,只需把它乘以字模所占字节数即可找到该字符地点字库的方位,如8×16的ASCII字库,ASCII码在该字库的方位为ASCII×16。假如中文字符和ASCII码混合在相同,怎么委任它们呢?其实也很简略,ASCII码的最高位是0,而中文的最高位是1,因而当读取到的一个字节的最高位是0,则该字节为ASCII码,它的下一个字节与这个字节无关;当取得到的字节的最高位是1,则一共的是中文字符,而且该字节与它的下一个字节组合在一起一共完好的一个汉字。

编码规矩介绍完了,那么怎么翻开字库呢?咱们能够运用前人已做好的字库,然后像拜访一般文件相同翻开它。另一种办法是把字库变换成一个超大的数组,那么咱们就能够像操作数组相同读取字库了(在这儿,咱们运用的是这种办法)。

下面咱们就给出详细的实例,它能够显现恣意的中、英文字符串。这儿只给出主程序,需求调用的子程序与上面的相同。

#include “font_libs.h”//内有两个数组__HZK[ ]和__ASCII[ ]
//别离一共中文字符和ASCII字符
…………
…………
void Main(void)
{
unsigned char String[]=”我的博客是:http://blog.csdn.net/zhaocj”;
int length = sizeof(String);
int k,xx;
unsigned char qh,wh;
const unsigned char *mould;
LCD_Init();
rLCDCON1|=1;

Brush_Background(0xffffff);
for(k=0,xx=0;k{
if(String[k]&0x80)//中文字符
{
qh=String[k]-0xa0;//区号
wh=String[k+1]-0xa0;//位号
mould = & __HZK[ ( ( qh – 1 )*94 + wh- 1 )*32 ];
Draw_Text16(4+xx,100,0x0f,mould);
xx+=16;
k++;
}
else//ASCII码字符
{
mould = & __ASCII[String[k]*16];
Draw_ASCII(4+xx,100,0×0,mould);
xx+=8;
}
}
while(1)
{
;
}
}

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/fangan/biancheng/265497.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部